From 59bf6c4e0bbbd1e1e3c75c9a978da903aaef152a Mon Sep 17 00:00:00 2001 From: Joe Lanford Date: Wed, 18 Sep 2024 16:04:45 -0400 Subject: [PATCH] fix: make post-upgrade-e2e checks wait for ClusterCatalog to be reconciled Signed-off-by: Joe Lanford --- config/base/manager/manager.yaml | 5 +- go.mod | 2 +- .../core/clustercatalog_controller.go | 4 + test/upgrade/unpack_test.go | 76 +++++++++++++++++++ 4 files changed, 84 insertions(+), 3 deletions(-) diff --git a/config/base/manager/manager.yaml b/config/base/manager/manager.yaml index f6a177f5..e9475ffa 100644 --- a/config/base/manager/manager.yaml +++ b/config/base/manager/manager.yaml @@ -12,9 +12,10 @@ kind: Deployment metadata: name: controller-manager namespace: system + annotations: + kubectl.kubernetes.io/default-logs-container: manager labels: - app.kubernetes.io/part-of: olm - app.kubernetes.io/name: catalogd + control-plane: catalogd-controller-manager spec: selector: matchLabels: diff --git a/go.mod b/go.mod index 902c01b4..5a503278 100644 --- a/go.mod +++ b/go.mod @@ -17,6 +17,7 @@ require ( github.com/prometheus/client_golang v1.20.3 github.com/spf13/pflag v1.0.5 github.com/stretchr/testify v1.9.0 + k8s.io/api v0.31.1 k8s.io/apiextensions-apiserver v0.31.1 k8s.io/apimachinery v0.31.1 k8s.io/apiserver v0.31.1 @@ -156,7 +157,6 @@ require ( gopkg.in/warnings.v0 v0.1.2 // indirect gopkg.in/yaml.v2 v2.4.0 // indirect gopkg.in/yaml.v3 v3.0.1 // indirect - k8s.io/api v0.31.1 // indirect k8s.io/klog/v2 v2.130.1 // indirect k8s.io/kube-openapi v0.0.0-20240228011516-70dd3763d340 // indirect k8s.io/utils v0.0.0-20240711033017-18e509b52bc8 // indirect diff --git a/internal/controllers/core/clustercatalog_controller.go b/internal/controllers/core/clustercatalog_controller.go index 7c8ccbfe..09b20ade 100644 --- a/internal/controllers/core/clustercatalog_controller.go +++ b/internal/controllers/core/clustercatalog_controller.go @@ -63,6 +63,10 @@ type ClusterCatalogReconciler struct { // - https://pkg.go.dev/sigs.k8s.io/controller-runtime@v0.11.0/pkg/reconcile func (r *ClusterCatalogReconciler) Reconcile(ctx context.Context, req ctrl.Request) (ctrl.Result, error) { l := log.FromContext(ctx).WithName("catalogd-controller") + ctx = log.IntoContext(ctx, l) + + l.V(1).Info("reconcile starting") + defer l.V(1).Info("reconcile ending") existingCatsrc := v1alpha1.ClusterCatalog{} if err := r.Client.Get(ctx, req.NamespacedName, &existingCatsrc); err != nil { diff --git a/test/upgrade/unpack_test.go b/test/upgrade/unpack_test.go index 83dcaba4..92974d6a 100644 --- a/test/upgrade/unpack_test.go +++ b/test/upgrade/unpack_test.go @@ -1,16 +1,24 @@ package upgradee2e import ( + "bufio" "context" + "fmt" "os" + "strings" + "time" . "github.com/onsi/ginkgo/v2" . "github.com/onsi/gomega" "github.com/google/go-cmp/cmp" + appsv1 "k8s.io/api/apps/v1" + corev1 "k8s.io/api/core/v1" "k8s.io/apimachinery/pkg/api/meta" metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" + "k8s.io/apimachinery/pkg/labels" "k8s.io/apimachinery/pkg/types" + "sigs.k8s.io/controller-runtime/pkg/client" catalogd "github.com/operator-framework/catalogd/api/core/v1alpha1" "github.com/operator-framework/catalogd/test/e2e" @@ -20,6 +28,43 @@ var _ = Describe("ClusterCatalog Unpacking", func() { When("A ClusterCatalog is created", func() { It("Successfully unpacks catalog contents", func() { ctx := context.Background() + + var managerDeployment appsv1.Deployment + managerLabelSelector := labels.Set{"control-plane": "catalogd-controller-manager"} + By("Checking that the controller-manager deployment is updated") + Eventually(func(g Gomega) { + var managerDeployments appsv1.DeploymentList + err := c.List(ctx, &managerDeployments, client.MatchingLabels(managerLabelSelector), client.InNamespace("olmv1-system")) + g.Expect(err).ToNot(HaveOccurred()) + g.Expect(managerDeployments.Items).To(HaveLen(1)) + managerDeployment = managerDeployments.Items[0] + g.Expect(managerDeployment.Status.UpdatedReplicas).To(Equal(*managerDeployment.Spec.Replicas)) + g.Expect(managerDeployment.Status.Replicas).To(Equal(*managerDeployment.Spec.Replicas)) + g.Expect(managerDeployment.Status.AvailableReplicas).To(Equal(*managerDeployment.Spec.Replicas)) + g.Expect(managerDeployment.Status.ReadyReplicas).To(Equal(*managerDeployment.Spec.Replicas)) + }).Should(Succeed()) + + var managerPod corev1.Pod + By("Waiting for only one controller-manager pod to remain") + Eventually(func(g Gomega) { + var managerPods corev1.PodList + err := c.List(ctx, &managerPods, client.MatchingLabels(managerLabelSelector)) + g.Expect(err).ToNot(HaveOccurred()) + g.Expect(managerPods.Items).To(HaveLen(1)) + managerPod = managerPods.Items[0] + }).Should(Succeed()) + + By("Reading logs to make sure that ClusterCatalog was reconciled by catalogd") + logCtx, cancel := context.WithTimeout(ctx, time.Minute) + defer cancel() + substrings := []string{ + "reconcile ending", + fmt.Sprintf(`"ClusterCatalog": {"name":"%s"}`, testClusterCatalogName), + } + found, err := watchPodLogsForSubstring(logCtx, &managerPod, "manager", substrings...) + Expect(err).ToNot(HaveOccurred()) + Expect(found).To(BeTrue()) + catalog := &catalogd.ClusterCatalog{} By("Ensuring ClusterCatalog has Status.Condition of Unpacked with a status == True") Eventually(func(g Gomega) { @@ -44,3 +89,34 @@ var _ = Describe("ClusterCatalog Unpacking", func() { }) }) }) + +func watchPodLogsForSubstring(ctx context.Context, pod *corev1.Pod, container string, substrings ...string) (bool, error) { + podLogOpts := corev1.PodLogOptions{ + Follow: true, + Container: container, + } + + req := kubeClient.CoreV1().Pods(pod.Namespace).GetLogs(pod.Name, &podLogOpts) + podLogs, err := req.Stream(ctx) + if err != nil { + return false, err + } + defer podLogs.Close() + + scanner := bufio.NewScanner(podLogs) + for scanner.Scan() { + line := scanner.Text() + + foundCount := 0 + for _, substring := range substrings { + if strings.Contains(line, substring) { + foundCount++ + } + } + if foundCount == len(substrings) { + return true, nil + } + } + + return false, scanner.Err() +}