deployments.go 6.9 KB

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