deployments.go 7.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197
  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 upgrades
  14. import (
  15. "context"
  16. "fmt"
  17. "time"
  18. appsv1 "k8s.io/api/apps/v1"
  19. metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
  20. "k8s.io/apimachinery/pkg/types"
  21. "k8s.io/apimachinery/pkg/util/wait"
  22. clientset "k8s.io/client-go/kubernetes"
  23. deploymentutil "k8s.io/kubernetes/pkg/controller/deployment/util"
  24. "k8s.io/kubernetes/test/e2e/framework"
  25. e2edeploy "k8s.io/kubernetes/test/e2e/framework/deployment"
  26. "k8s.io/kubernetes/test/e2e/upgrades"
  27. "github.com/onsi/ginkgo"
  28. imageutils "k8s.io/kubernetes/test/utils/image"
  29. )
  30. const (
  31. deploymentName = "dp"
  32. // poll is how often to poll pods, nodes and claims.
  33. poll = 2 * time.Second
  34. pollLongTimeout = 5 * time.Minute
  35. )
  36. // TODO: Test that the deployment stays available during master (and maybe
  37. // node and cluster upgrades).
  38. // DeploymentUpgradeTest tests that a deployment is using the same replica
  39. // sets before and after a cluster upgrade.
  40. type DeploymentUpgradeTest struct {
  41. oldDeploymentUID types.UID
  42. oldRSUID types.UID
  43. newRSUID types.UID
  44. }
  45. // Name returns the tracking name of the test.
  46. func (DeploymentUpgradeTest) Name() string { return "[sig-apps] deployment-upgrade" }
  47. // Setup creates a deployment and makes sure it has a new and an old replicaset running.
  48. func (t *DeploymentUpgradeTest) Setup(f *framework.Framework) {
  49. c := f.ClientSet
  50. nginxImage := imageutils.GetE2EImage(imageutils.Nginx)
  51. ns := f.Namespace.Name
  52. deploymentClient := c.AppsV1().Deployments(ns)
  53. rsClient := c.AppsV1().ReplicaSets(ns)
  54. ginkgo.By(fmt.Sprintf("Creating a deployment %q with 1 replica in namespace %q", deploymentName, ns))
  55. d := e2edeploy.NewDeployment(deploymentName, int32(1), map[string]string{"test": "upgrade"}, "nginx", nginxImage, appsv1.RollingUpdateDeploymentStrategyType)
  56. deployment, err := deploymentClient.Create(context.TODO(), d, metav1.CreateOptions{})
  57. framework.ExpectNoError(err)
  58. ginkgo.By(fmt.Sprintf("Waiting deployment %q to complete", deploymentName))
  59. framework.ExpectNoError(e2edeploy.WaitForDeploymentComplete(c, deployment))
  60. ginkgo.By(fmt.Sprintf("Getting replicaset revision 1 of deployment %q", deploymentName))
  61. rsSelector, err := metav1.LabelSelectorAsSelector(d.Spec.Selector)
  62. framework.ExpectNoError(err)
  63. rsList, err := rsClient.List(context.TODO(), metav1.ListOptions{LabelSelector: rsSelector.String()})
  64. framework.ExpectNoError(err)
  65. rss := rsList.Items
  66. if len(rss) != 1 {
  67. framework.ExpectNoError(fmt.Errorf("expected one replicaset, got %d", len(rss)))
  68. }
  69. t.oldRSUID = rss[0].UID
  70. ginkgo.By(fmt.Sprintf("Waiting for revision of the deployment %q to become 1", deploymentName))
  71. framework.ExpectNoError(waitForDeploymentRevision(c, deployment, "1"))
  72. // Trigger a new rollout so that we have some history.
  73. ginkgo.By(fmt.Sprintf("Triggering a new rollout for deployment %q", deploymentName))
  74. deployment, err = e2edeploy.UpdateDeploymentWithRetries(c, ns, deploymentName, func(update *appsv1.Deployment) {
  75. update.Spec.Template.Spec.Containers[0].Name = "updated-name"
  76. })
  77. framework.ExpectNoError(err)
  78. ginkgo.By(fmt.Sprintf("Waiting deployment %q to complete", deploymentName))
  79. framework.ExpectNoError(e2edeploy.WaitForDeploymentComplete(c, deployment))
  80. ginkgo.By(fmt.Sprintf("Getting replicasets revision 1 and 2 of deployment %q", deploymentName))
  81. rsList, err = rsClient.List(context.TODO(), metav1.ListOptions{LabelSelector: rsSelector.String()})
  82. framework.ExpectNoError(err)
  83. rss = rsList.Items
  84. if len(rss) != 2 {
  85. framework.ExpectNoError(fmt.Errorf("expected 2 replicaset, got %d", len(rss)))
  86. }
  87. ginkgo.By(fmt.Sprintf("Checking replicaset of deployment %q that is created before rollout survives the rollout", deploymentName))
  88. switch t.oldRSUID {
  89. case rss[0].UID:
  90. t.newRSUID = rss[1].UID
  91. case rss[1].UID:
  92. t.newRSUID = rss[0].UID
  93. default:
  94. framework.ExpectNoError(fmt.Errorf("old replicaset with UID %q does not survive rollout", t.oldRSUID))
  95. }
  96. ginkgo.By(fmt.Sprintf("Waiting for revision of the deployment %q to become 2", deploymentName))
  97. framework.ExpectNoError(waitForDeploymentRevision(c, deployment, "2"))
  98. t.oldDeploymentUID = deployment.UID
  99. }
  100. // Test checks whether the replicasets for a deployment are the same after an upgrade.
  101. func (t *DeploymentUpgradeTest) Test(f *framework.Framework, done <-chan struct{}, upgrade upgrades.UpgradeType) {
  102. // Block until upgrade is done
  103. ginkgo.By(fmt.Sprintf("Waiting for upgrade to finish before checking replicasets for deployment %q", deploymentName))
  104. <-done
  105. c := f.ClientSet
  106. ns := f.Namespace.Name
  107. deploymentClient := c.AppsV1().Deployments(ns)
  108. rsClient := c.AppsV1().ReplicaSets(ns)
  109. deployment, err := deploymentClient.Get(context.TODO(), deploymentName, metav1.GetOptions{})
  110. framework.ExpectNoError(err)
  111. ginkgo.By(fmt.Sprintf("Checking UID to verify deployment %q survives upgrade", deploymentName))
  112. framework.ExpectEqual(deployment.UID, t.oldDeploymentUID)
  113. ginkgo.By(fmt.Sprintf("Verifying deployment %q does not create new replicasets", deploymentName))
  114. rsSelector, err := metav1.LabelSelectorAsSelector(deployment.Spec.Selector)
  115. framework.ExpectNoError(err)
  116. rsList, err := rsClient.List(context.TODO(), metav1.ListOptions{LabelSelector: rsSelector.String()})
  117. framework.ExpectNoError(err)
  118. rss := rsList.Items
  119. if len(rss) != 2 {
  120. framework.ExpectNoError(fmt.Errorf("expected 2 replicaset, got %d", len(rss)))
  121. }
  122. switch t.oldRSUID {
  123. case rss[0].UID:
  124. framework.ExpectEqual(rss[1].UID, t.newRSUID)
  125. case rss[1].UID:
  126. framework.ExpectEqual(rss[0].UID, t.newRSUID)
  127. default:
  128. framework.ExpectNoError(fmt.Errorf("new replicasets are created during upgrade of deployment %q", deploymentName))
  129. }
  130. ginkgo.By(fmt.Sprintf("Verifying revision of the deployment %q is still 2", deploymentName))
  131. framework.ExpectEqual(deployment.Annotations[deploymentutil.RevisionAnnotation], "2")
  132. ginkgo.By(fmt.Sprintf("Waiting for deployment %q to complete adoption", deploymentName))
  133. framework.ExpectNoError(e2edeploy.WaitForDeploymentComplete(c, deployment))
  134. // Verify the upgraded deployment is active by scaling up the deployment by 1
  135. ginkgo.By(fmt.Sprintf("Scaling up replicaset of deployment %q by 1", deploymentName))
  136. deploymentWithUpdatedReplicas, err := e2edeploy.UpdateDeploymentWithRetries(c, ns, deploymentName, func(deployment *appsv1.Deployment) {
  137. *deployment.Spec.Replicas = *deployment.Spec.Replicas + 1
  138. })
  139. framework.ExpectNoError(err)
  140. ginkgo.By(fmt.Sprintf("Waiting for deployment %q to complete after scaling", deploymentName))
  141. framework.ExpectNoError(e2edeploy.WaitForDeploymentComplete(c, deploymentWithUpdatedReplicas))
  142. }
  143. // Teardown cleans up any remaining resources.
  144. func (t *DeploymentUpgradeTest) Teardown(f *framework.Framework) {
  145. // rely on the namespace deletion to clean up everything
  146. }
  147. // waitForDeploymentRevision waits for becoming the target revision of a delopyment.
  148. func waitForDeploymentRevision(c clientset.Interface, d *appsv1.Deployment, targetRevision string) error {
  149. err := wait.PollImmediate(poll, pollLongTimeout, func() (bool, error) {
  150. deployment, err := c.AppsV1().Deployments(d.Namespace).Get(context.TODO(), d.Name, metav1.GetOptions{})
  151. if err != nil {
  152. return false, err
  153. }
  154. revision := deployment.Annotations[deploymentutil.RevisionAnnotation]
  155. return revision == targetRevision, nil
  156. })
  157. if err != nil {
  158. return fmt.Errorf("error waiting for revision to become %q for deployment %q: %v", targetRevision, d.Name, err)
  159. }
  160. return nil
  161. }