resize_nodes.go 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172
  1. /*
  2. Copyright 2015 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 gcp
  14. import (
  15. "context"
  16. "fmt"
  17. "strings"
  18. "time"
  19. metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
  20. clientset "k8s.io/client-go/kubernetes"
  21. "k8s.io/kubernetes/test/e2e/common"
  22. "k8s.io/kubernetes/test/e2e/framework"
  23. e2enode "k8s.io/kubernetes/test/e2e/framework/node"
  24. e2epod "k8s.io/kubernetes/test/e2e/framework/pod"
  25. e2eskipper "k8s.io/kubernetes/test/e2e/framework/skipper"
  26. "github.com/onsi/ginkgo"
  27. )
  28. func resizeRC(c clientset.Interface, ns, name string, replicas int32) error {
  29. rc, err := c.CoreV1().ReplicationControllers(ns).Get(context.TODO(), name, metav1.GetOptions{})
  30. if err != nil {
  31. return err
  32. }
  33. *(rc.Spec.Replicas) = replicas
  34. _, err = c.CoreV1().ReplicationControllers(rc.Namespace).Update(context.TODO(), rc, metav1.UpdateOptions{})
  35. return err
  36. }
  37. var _ = SIGDescribe("Nodes [Disruptive]", func() {
  38. f := framework.NewDefaultFramework("resize-nodes")
  39. var systemPodsNo int32
  40. var c clientset.Interface
  41. var ns string
  42. var group string
  43. ginkgo.BeforeEach(func() {
  44. c = f.ClientSet
  45. ns = f.Namespace.Name
  46. systemPods, err := e2epod.GetPodsInNamespace(c, ns, map[string]string{})
  47. framework.ExpectNoError(err)
  48. systemPodsNo = int32(len(systemPods))
  49. if strings.Contains(framework.TestContext.CloudConfig.NodeInstanceGroup, ",") {
  50. framework.Failf("Test dose not support cluster setup with more than one MIG: %s", framework.TestContext.CloudConfig.NodeInstanceGroup)
  51. } else {
  52. group = framework.TestContext.CloudConfig.NodeInstanceGroup
  53. }
  54. })
  55. // Slow issue #13323 (8 min)
  56. ginkgo.Describe("Resize [Slow]", func() {
  57. var originalNodeCount int32
  58. var skipped bool
  59. ginkgo.BeforeEach(func() {
  60. skipped = true
  61. e2eskipper.SkipUnlessProviderIs("gce", "gke", "aws")
  62. e2eskipper.SkipUnlessNodeCountIsAtLeast(2)
  63. skipped = false
  64. })
  65. ginkgo.AfterEach(func() {
  66. if skipped {
  67. return
  68. }
  69. ginkgo.By("restoring the original node instance group size")
  70. if err := framework.ResizeGroup(group, int32(framework.TestContext.CloudConfig.NumNodes)); err != nil {
  71. framework.Failf("Couldn't restore the original node instance group size: %v", err)
  72. }
  73. // In GKE, our current tunneling setup has the potential to hold on to a broken tunnel (from a
  74. // rebooted/deleted node) for up to 5 minutes before all tunnels are dropped and recreated.
  75. // Most tests make use of some proxy feature to verify functionality. So, if a reboot test runs
  76. // right before a test that tries to get logs, for example, we may get unlucky and try to use a
  77. // closed tunnel to a node that was recently rebooted. There's no good way to framework.Poll for proxies
  78. // being closed, so we sleep.
  79. //
  80. // TODO(cjcullen) reduce this sleep (#19314)
  81. if framework.ProviderIs("gke") {
  82. ginkgo.By("waiting 5 minutes for all dead tunnels to be dropped")
  83. time.Sleep(5 * time.Minute)
  84. }
  85. if err := framework.WaitForGroupSize(group, int32(framework.TestContext.CloudConfig.NumNodes)); err != nil {
  86. framework.Failf("Couldn't restore the original node instance group size: %v", err)
  87. }
  88. if err := e2enode.WaitForReadyNodes(c, int(originalNodeCount), 10*time.Minute); err != nil {
  89. framework.Failf("Couldn't restore the original cluster size: %v", err)
  90. }
  91. // Many e2e tests assume that the cluster is fully healthy before they start. Wait until
  92. // the cluster is restored to health.
  93. ginkgo.By("waiting for system pods to successfully restart")
  94. err := e2epod.WaitForPodsRunningReady(c, metav1.NamespaceSystem, systemPodsNo, 0, framework.PodReadyBeforeTimeout, map[string]string{})
  95. framework.ExpectNoError(err)
  96. })
  97. ginkgo.It("should be able to delete nodes", func() {
  98. // Create a replication controller for a service that serves its hostname.
  99. // The source for the Docker container kubernetes/serve_hostname is in contrib/for-demos/serve_hostname
  100. name := "my-hostname-delete-node"
  101. numNodes, err := e2enode.TotalRegistered(c)
  102. framework.ExpectNoError(err)
  103. originalNodeCount = int32(numNodes)
  104. common.NewRCByName(c, ns, name, originalNodeCount, nil, nil)
  105. err = e2epod.VerifyPods(c, ns, name, true, originalNodeCount)
  106. framework.ExpectNoError(err)
  107. targetNumNodes := int32(framework.TestContext.CloudConfig.NumNodes - 1)
  108. ginkgo.By(fmt.Sprintf("decreasing cluster size to %d", targetNumNodes))
  109. err = framework.ResizeGroup(group, targetNumNodes)
  110. framework.ExpectNoError(err)
  111. err = framework.WaitForGroupSize(group, targetNumNodes)
  112. framework.ExpectNoError(err)
  113. err = e2enode.WaitForReadyNodes(c, int(originalNodeCount-1), 10*time.Minute)
  114. framework.ExpectNoError(err)
  115. ginkgo.By("waiting 1 minute for the watch in the podGC to catch up, remove any pods scheduled on " +
  116. "the now non-existent node and the RC to recreate it")
  117. time.Sleep(time.Minute)
  118. ginkgo.By("verifying whether the pods from the removed node are recreated")
  119. err = e2epod.VerifyPods(c, ns, name, true, originalNodeCount)
  120. framework.ExpectNoError(err)
  121. })
  122. // TODO: Bug here - testName is not correct
  123. ginkgo.It("should be able to add nodes", func() {
  124. // Create a replication controller for a service that serves its hostname.
  125. // The source for the Docker container kubernetes/serve_hostname is in contrib/for-demos/serve_hostname
  126. name := "my-hostname-add-node"
  127. common.NewSVCByName(c, ns, name)
  128. numNodes, err := e2enode.TotalRegistered(c)
  129. framework.ExpectNoError(err)
  130. originalNodeCount = int32(numNodes)
  131. common.NewRCByName(c, ns, name, originalNodeCount, nil, nil)
  132. err = e2epod.VerifyPods(c, ns, name, true, originalNodeCount)
  133. framework.ExpectNoError(err)
  134. targetNumNodes := int32(framework.TestContext.CloudConfig.NumNodes + 1)
  135. ginkgo.By(fmt.Sprintf("increasing cluster size to %d", targetNumNodes))
  136. err = framework.ResizeGroup(group, targetNumNodes)
  137. framework.ExpectNoError(err)
  138. err = framework.WaitForGroupSize(group, targetNumNodes)
  139. framework.ExpectNoError(err)
  140. err = e2enode.WaitForReadyNodes(c, int(originalNodeCount+1), 10*time.Minute)
  141. framework.ExpectNoError(err)
  142. ginkgo.By(fmt.Sprintf("increasing size of the replication controller to %d and verifying all pods are running", originalNodeCount+1))
  143. err = resizeRC(c, ns, name, originalNodeCount+1)
  144. framework.ExpectNoError(err)
  145. err = e2epod.VerifyPods(c, ns, name, true, originalNodeCount+1)
  146. framework.ExpectNoError(err)
  147. })
  148. })
  149. })