node_lease.go 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170
  1. /*
  2. Copyright 2018 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. v1 "k8s.io/api/core/v1"
  20. metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
  21. clientset "k8s.io/client-go/kubernetes"
  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. "github.com/onsi/gomega"
  28. )
  29. var _ = SIGDescribe("[Disruptive]NodeLease", func() {
  30. f := framework.NewDefaultFramework("node-lease-test")
  31. var systemPodsNo int32
  32. var c clientset.Interface
  33. var ns string
  34. var group string
  35. ginkgo.BeforeEach(func() {
  36. c = f.ClientSet
  37. ns = f.Namespace.Name
  38. systemPods, err := e2epod.GetPodsInNamespace(c, ns, map[string]string{})
  39. framework.ExpectNoError(err)
  40. systemPodsNo = int32(len(systemPods))
  41. if strings.Contains(framework.TestContext.CloudConfig.NodeInstanceGroup, ",") {
  42. framework.Failf("Test dose not support cluster setup with more than one MIG: %s", framework.TestContext.CloudConfig.NodeInstanceGroup)
  43. } else {
  44. group = framework.TestContext.CloudConfig.NodeInstanceGroup
  45. }
  46. })
  47. ginkgo.Describe("NodeLease deletion", func() {
  48. var skipped bool
  49. ginkgo.BeforeEach(func() {
  50. skipped = true
  51. e2eskipper.SkipUnlessProviderIs("gce", "gke", "aws")
  52. e2eskipper.SkipUnlessNodeCountIsAtLeast(2)
  53. skipped = false
  54. })
  55. ginkgo.AfterEach(func() {
  56. if skipped {
  57. return
  58. }
  59. ginkgo.By("restoring the original node instance group size")
  60. if err := framework.ResizeGroup(group, int32(framework.TestContext.CloudConfig.NumNodes)); err != nil {
  61. framework.Failf("Couldn't restore the original node instance group size: %v", err)
  62. }
  63. // In GKE, our current tunneling setup has the potential to hold on to a broken tunnel (from a
  64. // rebooted/deleted node) for up to 5 minutes before all tunnels are dropped and recreated.
  65. // Most tests make use of some proxy feature to verify functionality. So, if a reboot test runs
  66. // right before a test that tries to get logs, for example, we may get unlucky and try to use a
  67. // closed tunnel to a node that was recently rebooted. There's no good way to framework.Poll for proxies
  68. // being closed, so we sleep.
  69. //
  70. // TODO(cjcullen) reduce this sleep (#19314)
  71. if framework.ProviderIs("gke") {
  72. ginkgo.By("waiting 5 minutes for all dead tunnels to be dropped")
  73. time.Sleep(5 * time.Minute)
  74. }
  75. if err := framework.WaitForGroupSize(group, int32(framework.TestContext.CloudConfig.NumNodes)); err != nil {
  76. framework.Failf("Couldn't restore the original node instance group size: %v", err)
  77. }
  78. if err := e2enode.WaitForReadyNodes(c, framework.TestContext.CloudConfig.NumNodes, 10*time.Minute); err != nil {
  79. framework.Failf("Couldn't restore the original cluster size: %v", err)
  80. }
  81. // Many e2e tests assume that the cluster is fully healthy before they start. Wait until
  82. // the cluster is restored to health.
  83. ginkgo.By("waiting for system pods to successfully restart")
  84. err := e2epod.WaitForPodsRunningReady(c, metav1.NamespaceSystem, systemPodsNo, 0, framework.PodReadyBeforeTimeout, map[string]string{})
  85. framework.ExpectNoError(err)
  86. })
  87. ginkgo.It("node lease should be deleted when corresponding node is deleted", func() {
  88. leaseClient := c.CoordinationV1().Leases(v1.NamespaceNodeLease)
  89. err := e2enode.WaitForReadyNodes(c, framework.TestContext.CloudConfig.NumNodes, 10*time.Minute)
  90. framework.ExpectNoError(err)
  91. ginkgo.By("verify node lease exists for every nodes")
  92. originalNodes, err := e2enode.GetReadySchedulableNodes(c)
  93. framework.ExpectNoError(err)
  94. framework.ExpectEqual(len(originalNodes.Items), framework.TestContext.CloudConfig.NumNodes)
  95. gomega.Eventually(func() error {
  96. pass := true
  97. for _, node := range originalNodes.Items {
  98. if _, err := leaseClient.Get(context.TODO(), node.ObjectMeta.Name, metav1.GetOptions{}); err != nil {
  99. framework.Logf("Try to get lease of node %s, but got error: %v", node.ObjectMeta.Name, err)
  100. pass = false
  101. }
  102. }
  103. if pass {
  104. return nil
  105. }
  106. return fmt.Errorf("some node lease is not ready")
  107. }, 1*time.Minute, 5*time.Second).Should(gomega.BeNil())
  108. targetNumNodes := int32(framework.TestContext.CloudConfig.NumNodes - 1)
  109. ginkgo.By(fmt.Sprintf("decreasing cluster size to %d", targetNumNodes))
  110. err = framework.ResizeGroup(group, targetNumNodes)
  111. framework.ExpectNoError(err)
  112. err = framework.WaitForGroupSize(group, targetNumNodes)
  113. framework.ExpectNoError(err)
  114. err = e2enode.WaitForReadyNodes(c, framework.TestContext.CloudConfig.NumNodes-1, 10*time.Minute)
  115. framework.ExpectNoError(err)
  116. targetNodes, err := e2enode.GetReadySchedulableNodes(c)
  117. framework.ExpectNoError(err)
  118. framework.ExpectEqual(len(targetNodes.Items), int(targetNumNodes))
  119. ginkgo.By("verify node lease is deleted for the deleted node")
  120. var deletedNodeName string
  121. for _, originalNode := range originalNodes.Items {
  122. originalNodeName := originalNode.ObjectMeta.Name
  123. for _, targetNode := range targetNodes.Items {
  124. if originalNodeName == targetNode.ObjectMeta.Name {
  125. continue
  126. }
  127. }
  128. deletedNodeName = originalNodeName
  129. break
  130. }
  131. framework.ExpectNotEqual(deletedNodeName, "")
  132. gomega.Eventually(func() error {
  133. if _, err := leaseClient.Get(context.TODO(), deletedNodeName, metav1.GetOptions{}); err == nil {
  134. return fmt.Errorf("node lease is not deleted yet for node %q", deletedNodeName)
  135. }
  136. return nil
  137. }, 1*time.Minute, 5*time.Second).Should(gomega.BeNil())
  138. ginkgo.By("verify node leases still exist for remaining nodes")
  139. gomega.Eventually(func() error {
  140. for _, node := range targetNodes.Items {
  141. if _, err := leaseClient.Get(context.TODO(), node.ObjectMeta.Name, metav1.GetOptions{}); err != nil {
  142. return err
  143. }
  144. }
  145. return nil
  146. }, 1*time.Minute, 5*time.Second).Should(gomega.BeNil())
  147. })
  148. })
  149. })