From 21146ecd25a526cdbd5576d3d10e7d553b90d03d Mon Sep 17 00:00:00 2001 From: Warren Fernandes Date: Wed, 25 Mar 2020 15:52:23 -0600 Subject: [PATCH 1/6] Add test for etcd upgrade --- test/framework/workload_cluster.go | 74 +++++++++++++++++++ test/infrastructure/docker/e2e/docker_test.go | 31 ++++++++ 2 files changed, 105 insertions(+) create mode 100644 test/framework/workload_cluster.go diff --git a/test/framework/workload_cluster.go b/test/framework/workload_cluster.go new file mode 100644 index 000000000000..2d15b801f33c --- /dev/null +++ b/test/framework/workload_cluster.go @@ -0,0 +1,74 @@ +/* +Copyright 2019 The Kubernetes Authors. + +Licensed under the Apache License, Version 2.0 (the "License"); +you may not use this file except in compliance with the License. +You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + +Unless required by applicable law or agreed to in writing, software +distributed under the License is distributed on an "AS IS" BASIS, +WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +See the License for the specific language governing permissions and +limitations under the License. +*/ + +package framework + +import ( + "context" + "strings" + + . "github.com/onsi/ginkgo" + . "github.com/onsi/gomega" + "github.com/pkg/errors" + corev1 "k8s.io/api/core/v1" + "sigs.k8s.io/controller-runtime/pkg/client" +) + +// podCondition is a type that operates a condition on a Pod +type podCondition func(p *corev1.Pod) error + +// WaitForPodConditionInput is the input args for WaitForPodCondition +type WaitForPodConditionInput struct { + Lister Lister + ListOptions []client.ListOption + Condition func(p *corev1.Pod) error +} + +// WaitForPodCondition waits for the specified condition to be true for all +// pods returned from the list filter. +func WaitForPodCondition(ctx context.Context, input WaitForPodConditionInput, intervals ...interface{}) { + By("waiting for pod condition") + Eventually(func() (bool, error) { + podList := &corev1.PodList{} + if err := input.Lister.List( + ctx, + podList, + input.ListOptions..., + ); err != nil { + return false, err + } + satisfied := 0 + for _, pod := range podList.Items { + if input.Condition(&pod) == nil { + satisfied++ + } + } + // all pods in the list should satisfy the condition + return satisfied == len(podList.Items), nil + }, intervals...).Should(BeTrue()) +} + +// EtcdImageTagCondition returns a podCondition that ensures the pod image +// contains the specified image tag +func EtcdImageTagCondition(expectedTag string) podCondition { + return func(p *corev1.Pod) error { + // TODO: veriy container first + if !strings.Contains(p.Spec.Containers[0].Image, expectedTag) { + return errors.Errorf("pod %s/%s image does not contain the expectedTag %s", p.Namespace, p.Name, expectedTag) + } + return nil + } +} diff --git a/test/infrastructure/docker/e2e/docker_test.go b/test/infrastructure/docker/e2e/docker_test.go index a5c86bc6d06f..e3af2a235ab9 100644 --- a/test/infrastructure/docker/e2e/docker_test.go +++ b/test/infrastructure/docker/e2e/docker_test.go @@ -39,6 +39,7 @@ import ( infrav1 "sigs.k8s.io/cluster-api/test/infrastructure/docker/api/v1alpha3" "sigs.k8s.io/cluster-api/util" "sigs.k8s.io/cluster-api/util/patch" + "sigs.k8s.io/controller-runtime/pkg/client" ctrlclient "sigs.k8s.io/controller-runtime/pkg/client" ) @@ -181,6 +182,36 @@ var _ = Describe("Docker", func() { }, } framework.AssertControlPlaneFailureDomains(ctx, assertControlPlaneFailureDomainInput) + }) + + Specify("upgrades etcd", func() { + patchHelper, err := patch.NewHelper(controlPlane, mgmtClient) + Expect(err).ToNot(HaveOccurred()) + // This is the etcd version meant for k8s 1.17.x + // k8s 1.16.x clusters ususally get deployed with etcd 3.3.x + By("patching KubeadmConfigSpec etcd image tag") + controlPlane.Spec.KubeadmConfigSpec.ClusterConfiguration.Etcd = v1beta1.Etcd{ + Local: &v1beta1.LocalEtcd{ + ImageMeta: v1beta1.ImageMeta{ + ImageTag: "3.4.3-0", + }, + }, + } + Expect(patchHelper.Patch(ctx, controlPlane)).To(Succeed()) + // Get the etcd pods from the workload cluster and then verify + // the image of those pods are correct. + workloadClient, err := mgmt.GetWorkloadClient(ctx, cluster.Namespace, cluster.Name) + Expect(err).ToNot(HaveOccurred()) + + By("waiting for etcd pods to have the expected image tag") + waitForPodConditionInput := framework.WaitForPodConditionInput{ + Lister: workloadClient, + ListOptions: []client.ListOption{ + client.MatchingLabels(map[string]string{"component": "etcd"}), + }, + Condition: framework.EtcdImageTagCondition("3.4.3-0"), + } + framework.WaitForPodCondition(ctx, waitForPodConditionInput, "10m", "30s") }) From 56766360cd2c265dc1f0c3d1014eb614c46f9e67 Mon Sep 17 00:00:00 2001 From: Warren Fernandes Date: Wed, 25 Mar 2020 15:53:39 -0600 Subject: [PATCH 2/6] Remove outdated comments --- test/framework/control_plane.go | 2 -- 1 file changed, 2 deletions(-) diff --git a/test/framework/control_plane.go b/test/framework/control_plane.go index 1d88b979a3a4..21c3d533be86 100644 --- a/test/framework/control_plane.go +++ b/test/framework/control_plane.go @@ -286,8 +286,6 @@ type WaitForControlPlaneToBeReadyInput struct { } // WaitForControlPlaneToBeReady will wait for a control plane to be ready. -// TODO(chuckha): Once we implement control plane Ready, then we should update this to wait actually wait for ready. -// TODO(chuckha): In the meantime this uses initialized as a placeholder for Ready. func WaitForControlPlaneToBeReady(ctx context.Context, input WaitForControlPlaneToBeReadyInput, intervals ...interface{}) { By("waiting for the control plane to be ready") Eventually(func() (bool, error) { From 25dd126c4e4605a3f7a1aaa603ac66e4d9fef13e Mon Sep 17 00:00:00 2001 From: Warren Fernandes Date: Wed, 25 Mar 2020 21:00:24 -0600 Subject: [PATCH 3/6] Isolate upgrade etcd tests --- test/framework/workload_cluster.go | 80 +++++++---- test/infrastructure/docker/e2e/docker_test.go | 127 +++++++++++++++--- 2 files changed, 165 insertions(+), 42 deletions(-) diff --git a/test/framework/workload_cluster.go b/test/framework/workload_cluster.go index 2d15b801f33c..a7f128ba32b6 100644 --- a/test/framework/workload_cluster.go +++ b/test/framework/workload_cluster.go @@ -18,6 +18,7 @@ package framework import ( "context" + "fmt" "strings" . "github.com/onsi/ginkgo" @@ -27,47 +28,72 @@ import ( "sigs.k8s.io/controller-runtime/pkg/client" ) -// podCondition is a type that operates a condition on a Pod -type podCondition func(p *corev1.Pod) error +// podListCondition is a type that operates a condition on a Pod +type podListCondition func(p *corev1.PodList) error -// WaitForPodConditionInput is the input args for WaitForPodCondition -type WaitForPodConditionInput struct { +// WaitForPodListConditionInput is the input args for WaitForPodListCondition +type WaitForPodListConditionInput struct { Lister Lister - ListOptions []client.ListOption - Condition func(p *corev1.Pod) error + ListOptions *client.ListOptions + Condition podListCondition } -// WaitForPodCondition waits for the specified condition to be true for all +// WaitForPodListCondition waits for the specified condition to be true for all // pods returned from the list filter. -func WaitForPodCondition(ctx context.Context, input WaitForPodConditionInput, intervals ...interface{}) { - By("waiting for pod condition") +func WaitForPodListCondition(ctx context.Context, input WaitForPodListConditionInput, intervals ...interface{}) { Eventually(func() (bool, error) { podList := &corev1.PodList{} - if err := input.Lister.List( - ctx, - podList, - input.ListOptions..., - ); err != nil { + if err := input.Lister.List(ctx, podList, input.ListOptions); err != nil { return false, err } - satisfied := 0 - for _, pod := range podList.Items { - if input.Condition(&pod) == nil { - satisfied++ - } - } + Expect(len(podList.Items)).ToNot(BeZero()) + // all pods in the list should satisfy the condition - return satisfied == len(podList.Items), nil + err := input.Condition(podList) + if err != nil { + // DEBUG: + fmt.Println("---", err.Error()) + return false, err + } + return true, nil }, intervals...).Should(BeTrue()) + By("pod condition satisfied") } -// EtcdImageTagCondition returns a podCondition that ensures the pod image +// EtcdImageTagCondition returns a podListCondition that ensures the pod image // contains the specified image tag -func EtcdImageTagCondition(expectedTag string) podCondition { - return func(p *corev1.Pod) error { - // TODO: veriy container first - if !strings.Contains(p.Spec.Containers[0].Image, expectedTag) { - return errors.Errorf("pod %s/%s image does not contain the expectedTag %s", p.Namespace, p.Name, expectedTag) +func EtcdImageTagCondition(expectedTag string, expectedCount int) podListCondition { + return func(pl *corev1.PodList) error { + countWithCorrectTag := 0 + for _, pod := range pl.Items { + if strings.Contains(pod.Spec.Containers[0].Image, expectedTag) { + countWithCorrectTag++ + } + } + if countWithCorrectTag != expectedCount { + return errors.Errorf("expected %d pods to have image tag %q, got %d", expectedCount, expectedTag, countWithCorrectTag) + } + + // This check is to ensure that if there are three controlplane nodes, + // then there are only three etcd pods running. Currently, we create a + // new etcd pod before deleting the previous one. So we can have a + // case where there are three etcd pods with the correct tag and one + // left over that has yet to be deleted. + if len(pl.Items) != expectedCount { + return errors.Errorf("expected %d pods, got %d", expectedCount, len(pl.Items)) + } + return nil + } +} + +// PhasePodCondition is a podListCondition ensuring that pods are in the expected +// pod phase +func PhasePodCondition(expectedPhase corev1.PodPhase) podListCondition { + return func(pl *corev1.PodList) error { + for _, pod := range pl.Items { + if pod.Status.Phase != expectedPhase { + return errors.Errorf("pod %q is not %s", pod.Name, expectedPhase) + } } return nil } diff --git a/test/infrastructure/docker/e2e/docker_test.go b/test/infrastructure/docker/e2e/docker_test.go index e3af2a235ab9..1327f26e8f95 100644 --- a/test/infrastructure/docker/e2e/docker_test.go +++ b/test/infrastructure/docker/e2e/docker_test.go @@ -26,6 +26,8 @@ import ( . "github.com/onsi/ginkgo" . "github.com/onsi/gomega" + "k8s.io/apimachinery/pkg/labels" + "k8s.io/apimachinery/pkg/runtime" appsv1 "k8s.io/api/apps/v1" corev1 "k8s.io/api/core/v1" @@ -185,34 +187,129 @@ var _ = Describe("Docker", func() { }) Specify("upgrades etcd", func() { + replicas := 3 + var ( + infraCluster *infrav1.DockerCluster + template *infrav1.DockerMachineTemplate + err error + ) + cluster, infraCluster, controlPlane, template = clusterGen.GenerateCluster(namespace, int32(replicas)) + md, infraTemplate, bootstrapTemplate := GenerateMachineDeployment(cluster, 1) + + // Set up the client to the management cluster + mgmtClient, err = mgmt.GetClient() + Expect(err).NotTo(HaveOccurred()) + + // Set up the cluster object + createClusterInput := framework.CreateClusterInput{ + Creator: mgmtClient, + Cluster: cluster, + InfraCluster: infraCluster, + } + framework.CreateCluster(ctx, createClusterInput) + + // Set up the KubeadmControlPlane + createKubeadmControlPlaneInput := framework.CreateKubeadmControlPlaneInput{ + Creator: mgmtClient, + ControlPlane: controlPlane, + MachineTemplate: template, + } + framework.CreateKubeadmControlPlane(ctx, createKubeadmControlPlaneInput) + + // Wait for the cluster to provision. + assertClusterProvisionsInput := framework.WaitForClusterToProvisionInput{ + Getter: mgmtClient, + Cluster: cluster, + } + framework.WaitForClusterToProvision(ctx, assertClusterProvisionsInput) + + // Wait for at least one control plane node to be ready + waitForOneKubeadmControlPlaneMachineToExistInput := framework.WaitForOneKubeadmControlPlaneMachineToExistInput{ + Lister: mgmtClient, + Cluster: cluster, + ControlPlane: controlPlane, + } + framework.WaitForOneKubeadmControlPlaneMachineToExist(ctx, waitForOneKubeadmControlPlaneMachineToExistInput, "5m") + + // Insatll a networking solution on the workload cluster + workloadClient, err := mgmt.GetWorkloadClient(ctx, cluster.Namespace, cluster.Name) + Expect(err).ToNot(HaveOccurred()) + applyYAMLURLInput := framework.ApplyYAMLURLInput{ + Client: workloadClient, + HTTPGetter: http.DefaultClient, + NetworkingURL: "https://docs.projectcalico.org/manifests/calico.yaml", + Scheme: mgmt.Scheme, + } + framework.ApplyYAMLURL(ctx, applyYAMLURLInput) + + // Wait for the controlplane nodes to exist + assertKubeadmControlPlaneNodesExistInput := framework.WaitForKubeadmControlPlaneMachinesToExistInput{ + Lister: mgmtClient, + Cluster: cluster, + ControlPlane: controlPlane, + } + framework.WaitForKubeadmControlPlaneMachinesToExist(ctx, assertKubeadmControlPlaneNodesExistInput, "10m", "10s") + + // Create the workload nodes + createMachineDeploymentinput := framework.CreateMachineDeploymentInput{ + Creator: mgmtClient, + MachineDeployment: md, + BootstrapConfigTemplate: bootstrapTemplate, + InfraMachineTemplate: infraTemplate, + } + framework.CreateMachineDeployment(ctx, createMachineDeploymentinput) + + // Wait for the workload nodes to exist + waitForMachineDeploymentNodesToExistInput := framework.WaitForMachineDeploymentNodesToExistInput{ + Lister: mgmtClient, + Cluster: cluster, + MachineDeployment: md, + } + framework.WaitForMachineDeploymentNodesToExist(ctx, waitForMachineDeploymentNodesToExistInput) + + // Wait for the control plane to be ready + waitForControlPlaneToBeReadyInput := framework.WaitForControlPlaneToBeReadyInput{ + Getter: mgmtClient, + ControlPlane: controlPlane, + } + framework.WaitForControlPlaneToBeReady(ctx, waitForControlPlaneToBeReadyInput) + + // Before patching ensure all pods are ready in workload + // cluster + By("waiting for workload cluster pods to be Running") + waitForPodListConditionInput := framework.WaitForPodListConditionInput{ + Lister: workloadClient, + ListOptions: &client.ListOptions{Namespace: metav1.NamespaceSystem}, + Condition: framework.PhasePodCondition(corev1.PodRunning), + } + framework.WaitForPodListCondition(ctx, waitForPodListConditionInput) + + By("patching KubeadmConfigSpec etcd image tag in the kubeadmControlPlane") patchHelper, err := patch.NewHelper(controlPlane, mgmtClient) Expect(err).ToNot(HaveOccurred()) - // This is the etcd version meant for k8s 1.17.x - // k8s 1.16.x clusters ususally get deployed with etcd 3.3.x - By("patching KubeadmConfigSpec etcd image tag") controlPlane.Spec.KubeadmConfigSpec.ClusterConfiguration.Etcd = v1beta1.Etcd{ Local: &v1beta1.LocalEtcd{ ImageMeta: v1beta1.ImageMeta{ + // TODO: Ensure that the current version of etcd + // is not 3.4.3-0 or that k8s version is 1.16 + // 3.4.3-0 is the etcd version meant for k8s 1.17.x + // k8s 1.16.x clusters ususally get deployed with etcd 3.3.x ImageTag: "3.4.3-0", }, }, } Expect(patchHelper.Patch(ctx, controlPlane)).To(Succeed()) - // Get the etcd pods from the workload cluster and then verify - // the image of those pods are correct. - workloadClient, err := mgmt.GetWorkloadClient(ctx, cluster.Namespace, cluster.Name) - Expect(err).ToNot(HaveOccurred()) By("waiting for etcd pods to have the expected image tag") - waitForPodConditionInput := framework.WaitForPodConditionInput{ - Lister: workloadClient, - ListOptions: []client.ListOption{ - client.MatchingLabels(map[string]string{"component": "etcd"}), - }, - Condition: framework.EtcdImageTagCondition("3.4.3-0"), + lblSelector, err := labels.Parse("component=etcd") + Expect(err).ToNot(HaveOccurred()) + opt := &client.ListOptions{LabelSelector: lblSelector} + waitForPodListConditionInput = framework.WaitForPodListConditionInput{ + Lister: workloadClient, + ListOptions: opt, + Condition: framework.EtcdImageTagCondition("3.4.3-0", replicas), } - framework.WaitForPodCondition(ctx, waitForPodConditionInput, "10m", "30s") - + framework.WaitForPodListCondition(ctx, waitForPodListConditionInput) }) Specify("Full upgrade", func() { From 314a00b9fdc266685d1440442cd30d2372fe0420 Mon Sep 17 00:00:00 2001 From: Warren Fernandes Date: Wed, 25 Mar 2020 21:12:49 -0600 Subject: [PATCH 4/6] Delete workload cluster in AfterEach Since we are creating the workload clusters within the tests, let's delete them in the AfterEach instead of making the "mgmtClient" and "cluster" vars global. --- .../docker/e2e/docker_suite_test.go | 23 -------------- test/infrastructure/docker/e2e/docker_test.go | 30 +++++++++++++++++-- 2 files changed, 27 insertions(+), 26 deletions(-) diff --git a/test/infrastructure/docker/e2e/docker_suite_test.go b/test/infrastructure/docker/e2e/docker_suite_test.go index afb894931824..154160a64ec0 100644 --- a/test/infrastructure/docker/e2e/docker_suite_test.go +++ b/test/infrastructure/docker/e2e/docker_suite_test.go @@ -113,29 +113,6 @@ var _ = BeforeSuite(func() { }) var _ = AfterSuite(func() { - deleteClusterInput := framework.DeleteClusterInput{ - Deleter: mgmtClient, - Cluster: cluster, - } - framework.DeleteCluster(ctx, deleteClusterInput) - - waitForClusterDeletedInput := framework.WaitForClusterDeletedInput{ - Getter: mgmtClient, - Cluster: cluster, - } - framework.WaitForClusterDeleted(ctx, waitForClusterDeletedInput) - - assertAllClusterAPIResourcesAreGoneInput := framework.AssertAllClusterAPIResourcesAreGoneInput{ - Lister: mgmtClient, - Cluster: cluster, - } - framework.AssertAllClusterAPIResourcesAreGone(ctx, assertAllClusterAPIResourcesAreGoneInput) - - ensureDockerDeletedInput := ensureDockerArtifactsDeletedInput{ - Lister: mgmtClient, - Cluster: cluster, - } - ensureDockerArtifactsDeleted(ensureDockerDeletedInput) // Dump the logs of the providers before deleting them. Expect(writeLogs(mgmt, "capi-system", "capi-controller-manager", logPath)).To(Succeed()) diff --git a/test/infrastructure/docker/e2e/docker_test.go b/test/infrastructure/docker/e2e/docker_test.go index 1327f26e8f95..4d70509bf99e 100644 --- a/test/infrastructure/docker/e2e/docker_test.go +++ b/test/infrastructure/docker/e2e/docker_test.go @@ -45,15 +45,14 @@ import ( ctrlclient "sigs.k8s.io/controller-runtime/pkg/client" ) -var mgmtClient ctrlclient.Client -var cluster *clusterv1.Cluster - var _ = Describe("Docker", func() { Describe("Cluster Creation", func() { var ( namespace string clusterGen = &ClusterGenerator{} workloadClient ctrlclient.Client + mgmtClient ctrlclient.Client + cluster *clusterv1.Cluster ) SetDefaultEventuallyTimeout(10 * time.Minute) SetDefaultEventuallyPollingInterval(10 * time.Second) @@ -63,6 +62,31 @@ var _ = Describe("Docker", func() { }) AfterEach(func() { + // Delete the workload cluster + deleteClusterInput := framework.DeleteClusterInput{ + Deleter: mgmtClient, + Cluster: cluster, + } + framework.DeleteCluster(ctx, deleteClusterInput) + + waitForClusterDeletedInput := framework.WaitForClusterDeletedInput{ + Getter: mgmtClient, + Cluster: cluster, + } + framework.WaitForClusterDeleted(ctx, waitForClusterDeletedInput) + + assertAllClusterAPIResourcesAreGoneInput := framework.AssertAllClusterAPIResourcesAreGoneInput{ + Lister: mgmtClient, + Cluster: cluster, + } + framework.AssertAllClusterAPIResourcesAreGone(ctx, assertAllClusterAPIResourcesAreGoneInput) + + ensureDockerDeletedInput := ensureDockerArtifactsDeletedInput{ + Lister: mgmtClient, + Cluster: cluster, + } + ensureDockerArtifactsDeleted(ensureDockerDeletedInput) + // Dump cluster API and docker related resources to artifacts before deleting them. Expect(framework.DumpResources(mgmt, resourcesPath, GinkgoWriter)).To(Succeed()) resources := map[string]runtime.Object{ From e977a76b98e8b775e7f13059e5f3952bc6e6514b Mon Sep 17 00:00:00 2001 From: Warren Fernandes Date: Wed, 25 Mar 2020 22:01:52 -0600 Subject: [PATCH 5/6] Move etcd upgrade test into separate file Also pass in prefix to cluster generator and improve test output --- test/framework/workload_cluster.go | 6 +- test/infrastructure/docker/e2e/docker_test.go | 156 ++----------- .../docker/e2e/docker_upgrade_test.go | 220 ++++++++++++++++++ 3 files changed, 242 insertions(+), 140 deletions(-) create mode 100644 test/infrastructure/docker/e2e/docker_upgrade_test.go diff --git a/test/framework/workload_cluster.go b/test/framework/workload_cluster.go index a7f128ba32b6..43d14aa2bda1 100644 --- a/test/framework/workload_cluster.go +++ b/test/framework/workload_cluster.go @@ -52,7 +52,7 @@ func WaitForPodListCondition(ctx context.Context, input WaitForPodListConditionI err := input.Condition(podList) if err != nil { // DEBUG: - fmt.Println("---", err.Error()) + fmt.Println(err.Error()) return false, err } return true, nil @@ -71,7 +71,7 @@ func EtcdImageTagCondition(expectedTag string, expectedCount int) podListConditi } } if countWithCorrectTag != expectedCount { - return errors.Errorf("expected %d pods to have image tag %q, got %d", expectedCount, expectedTag, countWithCorrectTag) + return errors.Errorf("etcdImageTagCondition: expected %d pods to have image tag %q, got %d", expectedCount, expectedTag, countWithCorrectTag) } // This check is to ensure that if there are three controlplane nodes, @@ -80,7 +80,7 @@ func EtcdImageTagCondition(expectedTag string, expectedCount int) podListConditi // case where there are three etcd pods with the correct tag and one // left over that has yet to be deleted. if len(pl.Items) != expectedCount { - return errors.Errorf("expected %d pods, got %d", expectedCount, len(pl.Items)) + return errors.Errorf("etcdImageTagCondition: expected %d pods, got %d", expectedCount, len(pl.Items)) } return nil } diff --git a/test/infrastructure/docker/e2e/docker_test.go b/test/infrastructure/docker/e2e/docker_test.go index 4d70509bf99e..ef1c241b7908 100644 --- a/test/infrastructure/docker/e2e/docker_test.go +++ b/test/infrastructure/docker/e2e/docker_test.go @@ -26,7 +26,6 @@ import ( . "github.com/onsi/ginkgo" . "github.com/onsi/gomega" - "k8s.io/apimachinery/pkg/labels" "k8s.io/apimachinery/pkg/runtime" appsv1 "k8s.io/api/apps/v1" @@ -41,15 +40,14 @@ import ( infrav1 "sigs.k8s.io/cluster-api/test/infrastructure/docker/api/v1alpha3" "sigs.k8s.io/cluster-api/util" "sigs.k8s.io/cluster-api/util/patch" - "sigs.k8s.io/controller-runtime/pkg/client" ctrlclient "sigs.k8s.io/controller-runtime/pkg/client" ) var _ = Describe("Docker", func() { Describe("Cluster Creation", func() { var ( - namespace string - clusterGen = &ClusterGenerator{} + namespace = "default" + clusterGen = newClusterGenerator("") workloadClient ctrlclient.Client mgmtClient ctrlclient.Client cluster *clusterv1.Cluster @@ -57,10 +55,6 @@ var _ = Describe("Docker", func() { SetDefaultEventuallyTimeout(10 * time.Minute) SetDefaultEventuallyPollingInterval(10 * time.Second) - BeforeEach(func() { - namespace = "default" - }) - AfterEach(func() { // Delete the workload cluster deleteClusterInput := framework.DeleteClusterInput{ @@ -210,132 +204,6 @@ var _ = Describe("Docker", func() { framework.AssertControlPlaneFailureDomains(ctx, assertControlPlaneFailureDomainInput) }) - Specify("upgrades etcd", func() { - replicas := 3 - var ( - infraCluster *infrav1.DockerCluster - template *infrav1.DockerMachineTemplate - err error - ) - cluster, infraCluster, controlPlane, template = clusterGen.GenerateCluster(namespace, int32(replicas)) - md, infraTemplate, bootstrapTemplate := GenerateMachineDeployment(cluster, 1) - - // Set up the client to the management cluster - mgmtClient, err = mgmt.GetClient() - Expect(err).NotTo(HaveOccurred()) - - // Set up the cluster object - createClusterInput := framework.CreateClusterInput{ - Creator: mgmtClient, - Cluster: cluster, - InfraCluster: infraCluster, - } - framework.CreateCluster(ctx, createClusterInput) - - // Set up the KubeadmControlPlane - createKubeadmControlPlaneInput := framework.CreateKubeadmControlPlaneInput{ - Creator: mgmtClient, - ControlPlane: controlPlane, - MachineTemplate: template, - } - framework.CreateKubeadmControlPlane(ctx, createKubeadmControlPlaneInput) - - // Wait for the cluster to provision. - assertClusterProvisionsInput := framework.WaitForClusterToProvisionInput{ - Getter: mgmtClient, - Cluster: cluster, - } - framework.WaitForClusterToProvision(ctx, assertClusterProvisionsInput) - - // Wait for at least one control plane node to be ready - waitForOneKubeadmControlPlaneMachineToExistInput := framework.WaitForOneKubeadmControlPlaneMachineToExistInput{ - Lister: mgmtClient, - Cluster: cluster, - ControlPlane: controlPlane, - } - framework.WaitForOneKubeadmControlPlaneMachineToExist(ctx, waitForOneKubeadmControlPlaneMachineToExistInput, "5m") - - // Insatll a networking solution on the workload cluster - workloadClient, err := mgmt.GetWorkloadClient(ctx, cluster.Namespace, cluster.Name) - Expect(err).ToNot(HaveOccurred()) - applyYAMLURLInput := framework.ApplyYAMLURLInput{ - Client: workloadClient, - HTTPGetter: http.DefaultClient, - NetworkingURL: "https://docs.projectcalico.org/manifests/calico.yaml", - Scheme: mgmt.Scheme, - } - framework.ApplyYAMLURL(ctx, applyYAMLURLInput) - - // Wait for the controlplane nodes to exist - assertKubeadmControlPlaneNodesExistInput := framework.WaitForKubeadmControlPlaneMachinesToExistInput{ - Lister: mgmtClient, - Cluster: cluster, - ControlPlane: controlPlane, - } - framework.WaitForKubeadmControlPlaneMachinesToExist(ctx, assertKubeadmControlPlaneNodesExistInput, "10m", "10s") - - // Create the workload nodes - createMachineDeploymentinput := framework.CreateMachineDeploymentInput{ - Creator: mgmtClient, - MachineDeployment: md, - BootstrapConfigTemplate: bootstrapTemplate, - InfraMachineTemplate: infraTemplate, - } - framework.CreateMachineDeployment(ctx, createMachineDeploymentinput) - - // Wait for the workload nodes to exist - waitForMachineDeploymentNodesToExistInput := framework.WaitForMachineDeploymentNodesToExistInput{ - Lister: mgmtClient, - Cluster: cluster, - MachineDeployment: md, - } - framework.WaitForMachineDeploymentNodesToExist(ctx, waitForMachineDeploymentNodesToExistInput) - - // Wait for the control plane to be ready - waitForControlPlaneToBeReadyInput := framework.WaitForControlPlaneToBeReadyInput{ - Getter: mgmtClient, - ControlPlane: controlPlane, - } - framework.WaitForControlPlaneToBeReady(ctx, waitForControlPlaneToBeReadyInput) - - // Before patching ensure all pods are ready in workload - // cluster - By("waiting for workload cluster pods to be Running") - waitForPodListConditionInput := framework.WaitForPodListConditionInput{ - Lister: workloadClient, - ListOptions: &client.ListOptions{Namespace: metav1.NamespaceSystem}, - Condition: framework.PhasePodCondition(corev1.PodRunning), - } - framework.WaitForPodListCondition(ctx, waitForPodListConditionInput) - - By("patching KubeadmConfigSpec etcd image tag in the kubeadmControlPlane") - patchHelper, err := patch.NewHelper(controlPlane, mgmtClient) - Expect(err).ToNot(HaveOccurred()) - controlPlane.Spec.KubeadmConfigSpec.ClusterConfiguration.Etcd = v1beta1.Etcd{ - Local: &v1beta1.LocalEtcd{ - ImageMeta: v1beta1.ImageMeta{ - // TODO: Ensure that the current version of etcd - // is not 3.4.3-0 or that k8s version is 1.16 - // 3.4.3-0 is the etcd version meant for k8s 1.17.x - // k8s 1.16.x clusters ususally get deployed with etcd 3.3.x - ImageTag: "3.4.3-0", - }, - }, - } - Expect(patchHelper.Patch(ctx, controlPlane)).To(Succeed()) - - By("waiting for etcd pods to have the expected image tag") - lblSelector, err := labels.Parse("component=etcd") - Expect(err).ToNot(HaveOccurred()) - opt := &client.ListOptions{LabelSelector: lblSelector} - waitForPodListConditionInput = framework.WaitForPodListConditionInput{ - Lister: workloadClient, - ListOptions: opt, - Condition: framework.EtcdImageTagCondition("3.4.3-0", replicas), - } - framework.WaitForPodListCondition(ctx, waitForPodListConditionInput) - }) - Specify("Full upgrade", func() { By("upgrading the control plane object to a new version") patchHelper, err := patch.NewHelper(controlPlane, mgmtClient) @@ -449,12 +317,26 @@ func GenerateMachineDeployment(cluster *clusterv1.Cluster, replicas int32) (*clu return machineDeployment, infraTemplate, bootstrap } -type ClusterGenerator struct { +type clusterGenerator struct { + prefix string counter int } -func (c *ClusterGenerator) GenerateCluster(namespace string, replicas int32) (*clusterv1.Cluster, *infrav1.DockerCluster, *controlplanev1.KubeadmControlPlane, *infrav1.DockerMachineTemplate) { - generatedName := fmt.Sprintf("test-%d", c.counter) +func newClusterGenerator(name string) *clusterGenerator { + var prefix string + if len(name) != 0 { + prefix = fmt.Sprintf("test-%s-", name) + } else { + prefix = "test-" + } + + return &clusterGenerator{ + prefix: prefix, + } +} + +func (c *clusterGenerator) GenerateCluster(namespace string, replicas int32) (*clusterv1.Cluster, *infrav1.DockerCluster, *controlplanev1.KubeadmControlPlane, *infrav1.DockerMachineTemplate) { + generatedName := fmt.Sprintf("%s%d", c.prefix, c.counter) c.counter++ version := "v1.16.3" diff --git a/test/infrastructure/docker/e2e/docker_upgrade_test.go b/test/infrastructure/docker/e2e/docker_upgrade_test.go new file mode 100644 index 000000000000..33eada6b7442 --- /dev/null +++ b/test/infrastructure/docker/e2e/docker_upgrade_test.go @@ -0,0 +1,220 @@ +// +build e2e + +/* +Copyright 2019 The Kubernetes Authors. + +Licensed under the Apache License, Version 2.0 (the "License"); +you may not use this file except in compliance with the License. +You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + +Unless required by applicable law or agreed to in writing, software +distributed under the License is distributed on an "AS IS" BASIS, +WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +See the License for the specific language governing permissions and +limitations under the License. +*/ + +package e2e + +import ( + "net/http" + "time" + + . "github.com/onsi/ginkgo" + . "github.com/onsi/gomega" + corev1 "k8s.io/api/core/v1" + metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" + "k8s.io/apimachinery/pkg/labels" + "k8s.io/apimachinery/pkg/runtime" + + clusterv1 "sigs.k8s.io/cluster-api/api/v1alpha3" + "sigs.k8s.io/cluster-api/bootstrap/kubeadm/types/v1beta1" + controlplanev1 "sigs.k8s.io/cluster-api/controlplane/kubeadm/api/v1alpha3" + "sigs.k8s.io/cluster-api/test/framework" + infrav1 "sigs.k8s.io/cluster-api/test/infrastructure/docker/api/v1alpha3" + "sigs.k8s.io/cluster-api/util/patch" + "sigs.k8s.io/controller-runtime/pkg/client" + ctrlclient "sigs.k8s.io/controller-runtime/pkg/client" +) + +var _ = Describe("Docker Upgrade", func() { + var ( + replicas = 3 + namespace = "default" + clusterGen = newClusterGenerator("upgrade") + mgmtClient ctrlclient.Client + cluster *clusterv1.Cluster + controlPlane *controlplanev1.KubeadmControlPlane + ) + SetDefaultEventuallyTimeout(10 * time.Minute) + SetDefaultEventuallyPollingInterval(10 * time.Second) + + BeforeEach(func() { + // Ensure multi-controlplane workload cluster is up and running + var ( + infraCluster *infrav1.DockerCluster + template *infrav1.DockerMachineTemplate + err error + ) + cluster, infraCluster, controlPlane, template = clusterGen.GenerateCluster(namespace, int32(replicas)) + md, infraTemplate, bootstrapTemplate := GenerateMachineDeployment(cluster, 1) + + // Set up the client to the management cluster + mgmtClient, err = mgmt.GetClient() + Expect(err).NotTo(HaveOccurred()) + + // Set up the cluster object + createClusterInput := framework.CreateClusterInput{ + Creator: mgmtClient, + Cluster: cluster, + InfraCluster: infraCluster, + } + framework.CreateCluster(ctx, createClusterInput) + + // Set up the KubeadmControlPlane + createKubeadmControlPlaneInput := framework.CreateKubeadmControlPlaneInput{ + Creator: mgmtClient, + ControlPlane: controlPlane, + MachineTemplate: template, + } + framework.CreateKubeadmControlPlane(ctx, createKubeadmControlPlaneInput) + + // Wait for the cluster to provision. + assertClusterProvisionsInput := framework.WaitForClusterToProvisionInput{ + Getter: mgmtClient, + Cluster: cluster, + } + framework.WaitForClusterToProvision(ctx, assertClusterProvisionsInput) + + // Wait for at least one control plane node to be ready + waitForOneKubeadmControlPlaneMachineToExistInput := framework.WaitForOneKubeadmControlPlaneMachineToExistInput{ + Lister: mgmtClient, + Cluster: cluster, + ControlPlane: controlPlane, + } + framework.WaitForOneKubeadmControlPlaneMachineToExist(ctx, waitForOneKubeadmControlPlaneMachineToExistInput, "5m") + + // Insatll a networking solution on the workload cluster + workloadClient, err := mgmt.GetWorkloadClient(ctx, cluster.Namespace, cluster.Name) + Expect(err).ToNot(HaveOccurred()) + applyYAMLURLInput := framework.ApplyYAMLURLInput{ + Client: workloadClient, + HTTPGetter: http.DefaultClient, + NetworkingURL: "https://docs.projectcalico.org/manifests/calico.yaml", + Scheme: mgmt.Scheme, + } + framework.ApplyYAMLURL(ctx, applyYAMLURLInput) + + // Wait for the controlplane nodes to exist + assertKubeadmControlPlaneNodesExistInput := framework.WaitForKubeadmControlPlaneMachinesToExistInput{ + Lister: mgmtClient, + Cluster: cluster, + ControlPlane: controlPlane, + } + framework.WaitForKubeadmControlPlaneMachinesToExist(ctx, assertKubeadmControlPlaneNodesExistInput, "10m", "10s") + + // Create the workload nodes + createMachineDeploymentinput := framework.CreateMachineDeploymentInput{ + Creator: mgmtClient, + MachineDeployment: md, + BootstrapConfigTemplate: bootstrapTemplate, + InfraMachineTemplate: infraTemplate, + } + framework.CreateMachineDeployment(ctx, createMachineDeploymentinput) + + // Wait for the workload nodes to exist + waitForMachineDeploymentNodesToExistInput := framework.WaitForMachineDeploymentNodesToExistInput{ + Lister: mgmtClient, + Cluster: cluster, + MachineDeployment: md, + } + framework.WaitForMachineDeploymentNodesToExist(ctx, waitForMachineDeploymentNodesToExistInput) + + // Wait for the control plane to be ready + waitForControlPlaneToBeReadyInput := framework.WaitForControlPlaneToBeReadyInput{ + Getter: mgmtClient, + ControlPlane: controlPlane, + } + framework.WaitForControlPlaneToBeReady(ctx, waitForControlPlaneToBeReadyInput) + }) + + AfterEach(func() { + // Delete the workload cluster + deleteClusterInput := framework.DeleteClusterInput{ + Deleter: mgmtClient, + Cluster: cluster, + } + framework.DeleteCluster(ctx, deleteClusterInput) + + waitForClusterDeletedInput := framework.WaitForClusterDeletedInput{ + Getter: mgmtClient, + Cluster: cluster, + } + framework.WaitForClusterDeleted(ctx, waitForClusterDeletedInput) + + assertAllClusterAPIResourcesAreGoneInput := framework.AssertAllClusterAPIResourcesAreGoneInput{ + Lister: mgmtClient, + Cluster: cluster, + } + framework.AssertAllClusterAPIResourcesAreGone(ctx, assertAllClusterAPIResourcesAreGoneInput) + + ensureDockerDeletedInput := ensureDockerArtifactsDeletedInput{ + Lister: mgmtClient, + Cluster: cluster, + } + ensureDockerArtifactsDeleted(ensureDockerDeletedInput) + + // Dump cluster API and docker related resources to artifacts before deleting them. + Expect(framework.DumpResources(mgmt, resourcesPath, GinkgoWriter)).To(Succeed()) + resources := map[string]runtime.Object{ + "DockerCluster": &infrav1.DockerClusterList{}, + "DockerMachine": &infrav1.DockerMachineList{}, + "DockerMachineTemplate": &infrav1.DockerMachineTemplateList{}, + } + Expect(framework.DumpProviderResources(mgmt, resources, resourcesPath, GinkgoWriter)).To(Succeed()) + }) + + It("upgrades etcd", func() { + // Before patching ensure all pods are ready in workload + // cluster + workloadClient, err := mgmt.GetWorkloadClient(ctx, cluster.Namespace, cluster.Name) + Expect(err).ToNot(HaveOccurred()) + By("waiting for workload cluster pods to be Running") + waitForPodListConditionInput := framework.WaitForPodListConditionInput{ + Lister: workloadClient, + ListOptions: &client.ListOptions{Namespace: metav1.NamespaceSystem}, + Condition: framework.PhasePodCondition(corev1.PodRunning), + } + framework.WaitForPodListCondition(ctx, waitForPodListConditionInput) + + By("patching KubeadmConfigSpec etcd image tag in the kubeadmControlPlane") + patchHelper, err := patch.NewHelper(controlPlane, mgmtClient) + Expect(err).ToNot(HaveOccurred()) + controlPlane.Spec.KubeadmConfigSpec.ClusterConfiguration.Etcd = v1beta1.Etcd{ + Local: &v1beta1.LocalEtcd{ + ImageMeta: v1beta1.ImageMeta{ + // TODO: Ensure that the current version of etcd + // is not 3.4.3-0 or that k8s version is 1.16. For now it + // is. + // 3.4.3-0 is the etcd version meant for k8s 1.17.x + // k8s 1.16.x clusters ususally get deployed with etcd 3.3.x + ImageTag: "3.4.3-0", + }, + }, + } + Expect(patchHelper.Patch(ctx, controlPlane)).To(Succeed()) + + By("waiting for etcd pods to have the expected image tag") + lblSelector, err := labels.Parse("component=etcd") + Expect(err).ToNot(HaveOccurred()) + opt := &client.ListOptions{LabelSelector: lblSelector} + waitForPodListConditionInput = framework.WaitForPodListConditionInput{ + Lister: workloadClient, + ListOptions: opt, + Condition: framework.EtcdImageTagCondition("3.4.3-0", replicas), + } + framework.WaitForPodListCondition(ctx, waitForPodListConditionInput) + }) +}) From d205b684c57b6aa228c07c859746d9168433022e Mon Sep 17 00:00:00 2001 From: Warren Fernandes Date: Wed, 25 Mar 2020 23:15:59 -0600 Subject: [PATCH 6/6] Refactor upgrade tests --- test/infrastructure/docker/Makefile | 2 +- test/infrastructure/docker/e2e/docker_test.go | 363 ++++++++---------- .../docker/e2e/docker_upgrade_test.go | 75 +++- 3 files changed, 215 insertions(+), 225 deletions(-) diff --git a/test/infrastructure/docker/Makefile b/test/infrastructure/docker/Makefile index 198535aab7b6..f217393ddd5b 100644 --- a/test/infrastructure/docker/Makefile +++ b/test/infrastructure/docker/Makefile @@ -92,7 +92,7 @@ test-e2e: ## Run the end-to-end tests E2E_CONF_FILE ?= e2e/local-e2e.conf SKIP_RESOURCE_CLEANUP ?= false run-e2e: - go test ./e2e -v -ginkgo.v -ginkgo.trace -count=1 -timeout=20m -tags=e2e -e2e.config="$(abspath $(E2E_CONF_FILE))" -skip-resource-cleanup=$(SKIP_RESOURCE_CLEANUP) + go test ./e2e -v -ginkgo.v -ginkgo.trace -count=1 -timeout=35m -tags=e2e -e2e.config="$(abspath $(E2E_CONF_FILE))" -skip-resource-cleanup=$(SKIP_RESOURCE_CLEANUP) ## -------------------------------------- ## Binaries diff --git a/test/infrastructure/docker/e2e/docker_test.go b/test/infrastructure/docker/e2e/docker_test.go index ef1c241b7908..7cf27824cb08 100644 --- a/test/infrastructure/docker/e2e/docker_test.go +++ b/test/infrastructure/docker/e2e/docker_test.go @@ -19,7 +19,6 @@ limitations under the License. package e2e import ( - "errors" "fmt" "net/http" "time" @@ -28,10 +27,8 @@ import ( . "github.com/onsi/gomega" "k8s.io/apimachinery/pkg/runtime" - appsv1 "k8s.io/api/apps/v1" corev1 "k8s.io/api/core/v1" metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" - "k8s.io/apimachinery/pkg/runtime" clusterv1 "sigs.k8s.io/cluster-api/api/v1alpha3" bootstrapv1 "sigs.k8s.io/cluster-api/bootstrap/kubeadm/api/v1alpha3" "sigs.k8s.io/cluster-api/bootstrap/kubeadm/types/v1beta1" @@ -39,218 +36,166 @@ import ( "sigs.k8s.io/cluster-api/test/framework" infrav1 "sigs.k8s.io/cluster-api/test/infrastructure/docker/api/v1alpha3" "sigs.k8s.io/cluster-api/util" - "sigs.k8s.io/cluster-api/util/patch" ctrlclient "sigs.k8s.io/controller-runtime/pkg/client" ) -var _ = Describe("Docker", func() { - Describe("Cluster Creation", func() { +var _ = Describe("Docker Create", func() { + var ( + namespace = "default" + clusterGen = newClusterGenerator("create") + mgmtClient ctrlclient.Client + cluster *clusterv1.Cluster + ) + SetDefaultEventuallyTimeout(10 * time.Minute) + SetDefaultEventuallyPollingInterval(10 * time.Second) + + AfterEach(func() { + // Delete the workload cluster + deleteClusterInput := framework.DeleteClusterInput{ + Deleter: mgmtClient, + Cluster: cluster, + } + framework.DeleteCluster(ctx, deleteClusterInput) + + waitForClusterDeletedInput := framework.WaitForClusterDeletedInput{ + Getter: mgmtClient, + Cluster: cluster, + } + framework.WaitForClusterDeleted(ctx, waitForClusterDeletedInput) + + assertAllClusterAPIResourcesAreGoneInput := framework.AssertAllClusterAPIResourcesAreGoneInput{ + Lister: mgmtClient, + Cluster: cluster, + } + framework.AssertAllClusterAPIResourcesAreGone(ctx, assertAllClusterAPIResourcesAreGoneInput) + + ensureDockerDeletedInput := ensureDockerArtifactsDeletedInput{ + Lister: mgmtClient, + Cluster: cluster, + } + ensureDockerArtifactsDeleted(ensureDockerDeletedInput) + + // Dump cluster API and docker related resources to artifacts before deleting them. + Expect(framework.DumpResources(mgmt, resourcesPath, GinkgoWriter)).To(Succeed()) + resources := map[string]runtime.Object{ + "DockerCluster": &infrav1.DockerClusterList{}, + "DockerMachine": &infrav1.DockerMachineList{}, + "DockerMachineTemplate": &infrav1.DockerMachineTemplateList{}, + } + Expect(framework.DumpProviderResources(mgmt, resources, resourcesPath, GinkgoWriter)).To(Succeed()) + }) + + Specify("multi-node cluster with failure domains", func() { + replicas := 3 var ( - namespace = "default" - clusterGen = newClusterGenerator("") - workloadClient ctrlclient.Client - mgmtClient ctrlclient.Client - cluster *clusterv1.Cluster + infraCluster *infrav1.DockerCluster + template *infrav1.DockerMachineTemplate + controlPlane *controlplanev1.KubeadmControlPlane + err error ) - SetDefaultEventuallyTimeout(10 * time.Minute) - SetDefaultEventuallyPollingInterval(10 * time.Second) - - AfterEach(func() { - // Delete the workload cluster - deleteClusterInput := framework.DeleteClusterInput{ - Deleter: mgmtClient, - Cluster: cluster, - } - framework.DeleteCluster(ctx, deleteClusterInput) - - waitForClusterDeletedInput := framework.WaitForClusterDeletedInput{ - Getter: mgmtClient, - Cluster: cluster, - } - framework.WaitForClusterDeleted(ctx, waitForClusterDeletedInput) - - assertAllClusterAPIResourcesAreGoneInput := framework.AssertAllClusterAPIResourcesAreGoneInput{ - Lister: mgmtClient, - Cluster: cluster, - } - framework.AssertAllClusterAPIResourcesAreGone(ctx, assertAllClusterAPIResourcesAreGoneInput) - - ensureDockerDeletedInput := ensureDockerArtifactsDeletedInput{ - Lister: mgmtClient, - Cluster: cluster, - } - ensureDockerArtifactsDeleted(ensureDockerDeletedInput) - - // Dump cluster API and docker related resources to artifacts before deleting them. - Expect(framework.DumpResources(mgmt, resourcesPath, GinkgoWriter)).To(Succeed()) - resources := map[string]runtime.Object{ - "DockerCluster": &infrav1.DockerClusterList{}, - "DockerMachine": &infrav1.DockerMachineList{}, - "DockerMachineTemplate": &infrav1.DockerMachineTemplateList{}, - } - Expect(framework.DumpProviderResources(mgmt, resources, resourcesPath, GinkgoWriter)).To(Succeed()) - }) - - Describe("Multi-node controlplane cluster", func() { - var controlPlane *controlplanev1.KubeadmControlPlane - - Specify("Basic create", func() { - replicas := 3 - var ( - infraCluster *infrav1.DockerCluster - template *infrav1.DockerMachineTemplate - err error - ) - cluster, infraCluster, controlPlane, template = clusterGen.GenerateCluster(namespace, int32(replicas)) - // Set failure domains here - infraCluster.Spec.FailureDomains = clusterv1.FailureDomains{ - "domain-one": {ControlPlane: true}, - "domain-two": {ControlPlane: true}, - "domain-three": {ControlPlane: true}, - "domain-four": {ControlPlane: false}, - } - - md, infraTemplate, bootstrapTemplate := GenerateMachineDeployment(cluster, 1) - - // Set up the client to the management cluster - mgmtClient, err = mgmt.GetClient() - Expect(err).NotTo(HaveOccurred()) - - // Set up the cluster object - createClusterInput := framework.CreateClusterInput{ - Creator: mgmtClient, - Cluster: cluster, - InfraCluster: infraCluster, - } - framework.CreateCluster(ctx, createClusterInput) - - // Set up the KubeadmControlPlane - createKubeadmControlPlaneInput := framework.CreateKubeadmControlPlaneInput{ - Creator: mgmtClient, - ControlPlane: controlPlane, - MachineTemplate: template, - } - framework.CreateKubeadmControlPlane(ctx, createKubeadmControlPlaneInput) - - // Wait for the cluster to provision. - assertClusterProvisionsInput := framework.WaitForClusterToProvisionInput{ - Getter: mgmtClient, - Cluster: cluster, - } - framework.WaitForClusterToProvision(ctx, assertClusterProvisionsInput) - - // Wait for at least one control plane node to be ready - waitForOneKubeadmControlPlaneMachineToExistInput := framework.WaitForOneKubeadmControlPlaneMachineToExistInput{ - Lister: mgmtClient, - Cluster: cluster, - ControlPlane: controlPlane, - } - framework.WaitForOneKubeadmControlPlaneMachineToExist(ctx, waitForOneKubeadmControlPlaneMachineToExistInput, "5m") - - // Insatll a networking solution on the workload cluster - workloadClient, err = mgmt.GetWorkloadClient(ctx, cluster.Namespace, cluster.Name) - Expect(err).ToNot(HaveOccurred()) - applyYAMLURLInput := framework.ApplyYAMLURLInput{ - Client: workloadClient, - HTTPGetter: http.DefaultClient, - NetworkingURL: "https://docs.projectcalico.org/manifests/calico.yaml", - Scheme: mgmt.Scheme, - } - framework.ApplyYAMLURL(ctx, applyYAMLURLInput) - - // Wait for the controlplane nodes to exist - assertKubeadmControlPlaneNodesExistInput := framework.WaitForKubeadmControlPlaneMachinesToExistInput{ - Lister: mgmtClient, - Cluster: cluster, - ControlPlane: controlPlane, - } - framework.WaitForKubeadmControlPlaneMachinesToExist(ctx, assertKubeadmControlPlaneNodesExistInput, "10m", "10s") - - // Create the workload nodes - createMachineDeploymentinput := framework.CreateMachineDeploymentInput{ - Creator: mgmtClient, - MachineDeployment: md, - BootstrapConfigTemplate: bootstrapTemplate, - InfraMachineTemplate: infraTemplate, - } - framework.CreateMachineDeployment(ctx, createMachineDeploymentinput) - - // Wait for the workload nodes to exist - waitForMachineDeploymentNodesToExistInput := framework.WaitForMachineDeploymentNodesToExistInput{ - Lister: mgmtClient, - Cluster: cluster, - MachineDeployment: md, - } - framework.WaitForMachineDeploymentNodesToExist(ctx, waitForMachineDeploymentNodesToExistInput) - - // Wait for the control plane to be ready - waitForControlPlaneToBeReadyInput := framework.WaitForControlPlaneToBeReadyInput{ - Getter: mgmtClient, - ControlPlane: controlPlane, - } - framework.WaitForControlPlaneToBeReady(ctx, waitForControlPlaneToBeReadyInput) - - // Assert failure domain is working as expected - assertControlPlaneFailureDomainInput := framework.AssertControlPlaneFailureDomainsInput{ - GetLister: mgmtClient, - ClusterKey: util.ObjectKey(cluster), - ExpectedFailureDomains: map[string]int{ - "domain-one": 1, - "domain-two": 1, - "domain-three": 1, - "domain-four": 0, - }, - } - framework.AssertControlPlaneFailureDomains(ctx, assertControlPlaneFailureDomainInput) - }) - - Specify("Full upgrade", func() { - By("upgrading the control plane object to a new version") - patchHelper, err := patch.NewHelper(controlPlane, mgmtClient) - Expect(err).ToNot(HaveOccurred()) - controlPlane.Spec.Version = "v1.17.2" - Expect(patchHelper.Patch(ctx, controlPlane)).To(Succeed()) - By("waiting for all control plane nodes to exist") - inClustersNamespaceListOption := ctrlclient.InNamespace(cluster.Namespace) - // ControlPlane labels - matchClusterListOption := ctrlclient.MatchingLabels{ - clusterv1.MachineControlPlaneLabelName: "", - clusterv1.ClusterLabelName: cluster.Name, - } - - Eventually(func() (int, error) { - machineList := &clusterv1.MachineList{} - if err := mgmtClient.List(ctx, machineList, inClustersNamespaceListOption, matchClusterListOption); err != nil { - fmt.Println(err) - return 0, err - } - upgraded := 0 - for _, machine := range machineList.Items { - if *machine.Spec.Version == controlPlane.Spec.Version { - upgraded++ - } - } - if len(machineList.Items) > upgraded { - return 0, errors.New("old nodes remain") - } - return upgraded, nil - }, "10m", "30s").Should(Equal(int(*controlPlane.Spec.Replicas))) - Eventually(func() (bool, error) { - ds := &appsv1.DaemonSet{} - - if err := workloadClient.Get(ctx, ctrlclient.ObjectKey{Name: "kube-proxy", Namespace: metav1.NamespaceSystem}, ds); err != nil { - return false, err - } - if ds.Spec.Template.Spec.Containers[0].Image == "k8s.gcr.io/kube-proxy:v1.17.2" { - return true, nil - } - - return false, nil - }, "10m", "30s").Should(BeTrue()) - - }) - }) + cluster, infraCluster, controlPlane, template = clusterGen.GenerateCluster(namespace, int32(replicas)) + // Set failure domains here + infraCluster.Spec.FailureDomains = clusterv1.FailureDomains{ + "domain-one": {ControlPlane: true}, + "domain-two": {ControlPlane: true}, + "domain-three": {ControlPlane: true}, + "domain-four": {ControlPlane: false}, + } + + md, infraTemplate, bootstrapTemplate := GenerateMachineDeployment(cluster, 1) + + // Set up the client to the management cluster + mgmtClient, err = mgmt.GetClient() + Expect(err).NotTo(HaveOccurred()) + + // Set up the cluster object + createClusterInput := framework.CreateClusterInput{ + Creator: mgmtClient, + Cluster: cluster, + InfraCluster: infraCluster, + } + framework.CreateCluster(ctx, createClusterInput) + + // Set up the KubeadmControlPlane + createKubeadmControlPlaneInput := framework.CreateKubeadmControlPlaneInput{ + Creator: mgmtClient, + ControlPlane: controlPlane, + MachineTemplate: template, + } + framework.CreateKubeadmControlPlane(ctx, createKubeadmControlPlaneInput) + + // Wait for the cluster to provision. + assertClusterProvisionsInput := framework.WaitForClusterToProvisionInput{ + Getter: mgmtClient, + Cluster: cluster, + } + framework.WaitForClusterToProvision(ctx, assertClusterProvisionsInput) + + // Wait for at least one control plane node to be ready + waitForOneKubeadmControlPlaneMachineToExistInput := framework.WaitForOneKubeadmControlPlaneMachineToExistInput{ + Lister: mgmtClient, + Cluster: cluster, + ControlPlane: controlPlane, + } + framework.WaitForOneKubeadmControlPlaneMachineToExist(ctx, waitForOneKubeadmControlPlaneMachineToExistInput, "5m") + + // Insatll a networking solution on the workload cluster + workloadClient, err := mgmt.GetWorkloadClient(ctx, cluster.Namespace, cluster.Name) + Expect(err).ToNot(HaveOccurred()) + applyYAMLURLInput := framework.ApplyYAMLURLInput{ + Client: workloadClient, + HTTPGetter: http.DefaultClient, + NetworkingURL: "https://docs.projectcalico.org/manifests/calico.yaml", + Scheme: mgmt.Scheme, + } + framework.ApplyYAMLURL(ctx, applyYAMLURLInput) + + // Wait for the controlplane nodes to exist + assertKubeadmControlPlaneNodesExistInput := framework.WaitForKubeadmControlPlaneMachinesToExistInput{ + Lister: mgmtClient, + Cluster: cluster, + ControlPlane: controlPlane, + } + framework.WaitForKubeadmControlPlaneMachinesToExist(ctx, assertKubeadmControlPlaneNodesExistInput, "15m", "10s") + + // Create the workload nodes + createMachineDeploymentinput := framework.CreateMachineDeploymentInput{ + Creator: mgmtClient, + MachineDeployment: md, + BootstrapConfigTemplate: bootstrapTemplate, + InfraMachineTemplate: infraTemplate, + } + framework.CreateMachineDeployment(ctx, createMachineDeploymentinput) + + // Wait for the workload nodes to exist + waitForMachineDeploymentNodesToExistInput := framework.WaitForMachineDeploymentNodesToExistInput{ + Lister: mgmtClient, + Cluster: cluster, + MachineDeployment: md, + } + framework.WaitForMachineDeploymentNodesToExist(ctx, waitForMachineDeploymentNodesToExistInput) + + // Wait for the control plane to be ready + waitForControlPlaneToBeReadyInput := framework.WaitForControlPlaneToBeReadyInput{ + Getter: mgmtClient, + ControlPlane: controlPlane, + } + framework.WaitForControlPlaneToBeReady(ctx, waitForControlPlaneToBeReadyInput) + + // Assert failure domain is working as expected + assertControlPlaneFailureDomainInput := framework.AssertControlPlaneFailureDomainsInput{ + GetLister: mgmtClient, + ClusterKey: util.ObjectKey(cluster), + ExpectedFailureDomains: map[string]int{ + "domain-one": 1, + "domain-two": 1, + "domain-three": 1, + "domain-four": 0, + }, + } + framework.AssertControlPlaneFailureDomains(ctx, assertControlPlaneFailureDomainInput) }) + }) func GenerateMachineDeployment(cluster *clusterv1.Cluster, replicas int32) (*clusterv1.MachineDeployment, *infrav1.DockerMachineTemplate, *bootstrapv1.KubeadmConfigTemplate) { diff --git a/test/infrastructure/docker/e2e/docker_upgrade_test.go b/test/infrastructure/docker/e2e/docker_upgrade_test.go index 33eada6b7442..c59dfc589193 100644 --- a/test/infrastructure/docker/e2e/docker_upgrade_test.go +++ b/test/infrastructure/docker/e2e/docker_upgrade_test.go @@ -19,11 +19,14 @@ limitations under the License. package e2e import ( + "errors" + "fmt" "net/http" "time" . "github.com/onsi/ginkgo" . "github.com/onsi/gomega" + appsv1 "k8s.io/api/apps/v1" corev1 "k8s.io/api/core/v1" metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" "k8s.io/apimachinery/pkg/labels" @@ -176,22 +179,11 @@ var _ = Describe("Docker Upgrade", func() { Expect(framework.DumpProviderResources(mgmt, resources, resourcesPath, GinkgoWriter)).To(Succeed()) }) - It("upgrades etcd", func() { - // Before patching ensure all pods are ready in workload - // cluster - workloadClient, err := mgmt.GetWorkloadClient(ctx, cluster.Namespace, cluster.Name) - Expect(err).ToNot(HaveOccurred()) - By("waiting for workload cluster pods to be Running") - waitForPodListConditionInput := framework.WaitForPodListConditionInput{ - Lister: workloadClient, - ListOptions: &client.ListOptions{Namespace: metav1.NamespaceSystem}, - Condition: framework.PhasePodCondition(corev1.PodRunning), - } - framework.WaitForPodListCondition(ctx, waitForPodListConditionInput) - - By("patching KubeadmConfigSpec etcd image tag in the kubeadmControlPlane") + It("upgrades kubernetes, kube-proxy and etcd", func() { + By("upgrading kubernetes version and etcd image tag") patchHelper, err := patch.NewHelper(controlPlane, mgmtClient) Expect(err).ToNot(HaveOccurred()) + controlPlane.Spec.Version = "v1.17.2" controlPlane.Spec.KubeadmConfigSpec.ClusterConfiguration.Etcd = v1beta1.Etcd{ Local: &v1beta1.LocalEtcd{ ImageMeta: v1beta1.ImageMeta{ @@ -206,7 +198,60 @@ var _ = Describe("Docker Upgrade", func() { } Expect(patchHelper.Patch(ctx, controlPlane)).To(Succeed()) - By("waiting for etcd pods to have the expected image tag") + inClustersNamespaceListOption := ctrlclient.InNamespace(cluster.Namespace) + // ControlPlane labels + matchClusterListOption := ctrlclient.MatchingLabels{ + clusterv1.MachineControlPlaneLabelName: "", + clusterv1.ClusterLabelName: cluster.Name, + } + + By("ensuring all machines have upgraded kubernetes") + Eventually(func() (int, error) { + machineList := &clusterv1.MachineList{} + if err := mgmtClient.List(ctx, machineList, inClustersNamespaceListOption, matchClusterListOption); err != nil { + fmt.Println(err) + return 0, err + } + upgraded := 0 + for _, machine := range machineList.Items { + if *machine.Spec.Version == controlPlane.Spec.Version { + upgraded++ + } + } + if len(machineList.Items) > upgraded { + return 0, errors.New("old nodes remain") + } + return upgraded, nil + }, "10m", "30s").Should(Equal(int(*controlPlane.Spec.Replicas))) + + workloadClient, err := mgmt.GetWorkloadClient(ctx, cluster.Namespace, cluster.Name) + Expect(err).ToNot(HaveOccurred()) + + By("ensuring kube-proxy has the correct image") + Eventually(func() (bool, error) { + ds := &appsv1.DaemonSet{} + + if err := workloadClient.Get(ctx, ctrlclient.ObjectKey{Name: "kube-proxy", Namespace: metav1.NamespaceSystem}, ds); err != nil { + return false, err + } + if ds.Spec.Template.Spec.Containers[0].Image == "k8s.gcr.io/kube-proxy:v1.17.2" { + return true, nil + } + + return false, nil + }, "10m", "30s").Should(BeTrue()) + + // Before patching ensure all pods are ready in workload cluster + // Might not need this step any more. + By("waiting for workload cluster pods to be Running") + waitForPodListConditionInput := framework.WaitForPodListConditionInput{ + Lister: workloadClient, + ListOptions: &client.ListOptions{Namespace: metav1.NamespaceSystem}, + Condition: framework.PhasePodCondition(corev1.PodRunning), + } + framework.WaitForPodListCondition(ctx, waitForPodListConditionInput) + + By("ensuring etcd pods have the correct image tag") lblSelector, err := labels.Parse("component=etcd") Expect(err).ToNot(HaveOccurred()) opt := &client.ListOptions{LabelSelector: lblSelector}