From 2f0470f64a40dcafce8eca6d44acd6e54330e14b Mon Sep 17 00:00:00 2001 From: Muhammad Faizan Date: Mon, 11 Sep 2023 13:50:59 +0200 Subject: [PATCH] Added end-to-end setup tests (#83) * added e2e tests * added tests * added support for EventMesh * updated * updated * updated * addressed review comments * Update Makefile --- Makefile | 10 + api/v1alpha1/zz_generated.deepcopy.go | 2 +- hack/e2e/cleanup/cleanup_test.go | 244 +++++++++++++ hack/e2e/common/fixtures/fixtures.go | 121 +++++++ hack/e2e/common/k8s.go | 64 ++++ hack/e2e/common/logger.go | 36 ++ hack/e2e/common/retry.go | 38 +++ hack/e2e/env/env.go | 28 ++ hack/e2e/setup/setup_test.go | 475 ++++++++++++++++++++++++++ 9 files changed, 1017 insertions(+), 1 deletion(-) create mode 100644 hack/e2e/cleanup/cleanup_test.go create mode 100644 hack/e2e/common/fixtures/fixtures.go create mode 100644 hack/e2e/common/k8s.go create mode 100644 hack/e2e/common/logger.go create mode 100644 hack/e2e/common/retry.go create mode 100644 hack/e2e/env/env.go create mode 100644 hack/e2e/setup/setup_test.go diff --git a/Makefile b/Makefile index 0f869f62..cfb55598 100644 --- a/Makefile +++ b/Makefile @@ -186,3 +186,13 @@ fmt-local: ## Reformat files using `go fmt` imports-local: ## Optimize imports goimports -w -l $$($(FILES_TO_CHECK)) + +# e2e-setup will create an Eventing CR and check if the required resources are provisioned or not. +.PHONY: e2e-setup +e2e-setup: + go test -v ./hack/e2e/setup/setup_test.go --tags=e2e + +# e2e-cleanup will delete the Eventing CR and check if the required resources are de-provisioned or not. +.PHONY: e2e-cleanup +e2e-cleanup: + go test -v ./hack/e2e/cleanup/cleanup_test.go --tags=e2e diff --git a/api/v1alpha1/zz_generated.deepcopy.go b/api/v1alpha1/zz_generated.deepcopy.go index 28204a61..99b1cdc5 100644 --- a/api/v1alpha1/zz_generated.deepcopy.go +++ b/api/v1alpha1/zz_generated.deepcopy.go @@ -22,7 +22,7 @@ limitations under the License. package v1alpha1 import ( - "k8s.io/apimachinery/pkg/apis/meta/v1" + v1 "k8s.io/apimachinery/pkg/apis/meta/v1" runtime "k8s.io/apimachinery/pkg/runtime" ) diff --git a/hack/e2e/cleanup/cleanup_test.go b/hack/e2e/cleanup/cleanup_test.go new file mode 100644 index 00000000..291b0b52 --- /dev/null +++ b/hack/e2e/cleanup/cleanup_test.go @@ -0,0 +1,244 @@ +//go:build e2e +// +build e2e + +// Package cleanup-test is part of the end-to-end-tests. This package contains tests that evaluate the deletion of +// Eventing CRs and the cascading deletion of all correlated Kubernetes resources. +// To run the tests a k8s cluster and an Eventing-CR need to be available and configured. For this reason, the tests are +// seperated via the 'e2e' buildtags. For more information please consult the readme. +package cleanup_test + +import ( + "context" + "fmt" + "os" + "testing" + "time" + + eventingv1alpha1 "github.com/kyma-project/eventing-manager/api/v1alpha1" + "github.com/kyma-project/eventing-manager/hack/e2e/env" + "github.com/kyma-project/eventing-manager/pkg/eventing" + "github.com/pkg/errors" + "github.com/stretchr/testify/require" + k8serrors "k8s.io/apimachinery/pkg/api/errors" + metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" + + "go.uber.org/zap" + k8stypes "k8s.io/apimachinery/pkg/types" + + "k8s.io/client-go/kubernetes" + "sigs.k8s.io/controller-runtime/pkg/client" + + . "github.com/kyma-project/eventing-manager/hack/e2e/common" + . "github.com/kyma-project/eventing-manager/hack/e2e/common/fixtures" +) + +// Constants for retries. +const ( + interval = 2 * time.Second + attempts = 60 +) + +// clientSet is what is used to access K8s build-in resources like Pods, Namespaces and so on. +var clientSet *kubernetes.Clientset //nolint:gochecknoglobals // This will only be accessible in e2e tests. + +// k8sClient is what is used to access the Eventing CR. +var k8sClient client.Client //nolint:gochecknoglobals // This will only be accessible in e2e tests. + +var logger *zap.Logger + +var testConfigs *env.E2EConfig + +// TestMain runs before all the other test functions. It sets up all the resources that are shared between the different +// test functions. It will then run the tests and finally shuts everything down. +func TestMain(m *testing.M) { + var err error + logger, err = SetupLogger() + if err != nil { + logger.Fatal(err.Error()) + } + + testConfigs, err = env.GetE2EConfig() + if err != nil { + logger.Fatal(err.Error()) + + } + logger.Info(fmt.Sprintf("##### NOTE: Tests will run w.r.t. backend: %s", testConfigs.BackendType)) + + clientSet, k8sClient, err = GetK8sClients() + if err != nil { + logger.Fatal(err.Error()) + } + + ctx := context.TODO() + // Delete the Eventing CR used for testing. + err = Retry(attempts, interval, func() error { + return k8sClient.Delete(ctx, EventingCR(eventingv1alpha1.BackendType(testConfigs.BackendType))) + }) + if err != nil { + logger.Fatal(err.Error()) + } + + // Run the tests and exit. + code := m.Run() + os.Exit(code) +} + +func Test_NoEventingCRExists(t *testing.T) { + t.Parallel() + + ctx := context.TODO() + err := Retry(attempts, interval, func() error { + _, crErr := getEventingCR(ctx, CRName, NamespaceName) + // This is what we want here. + if k8serrors.IsNotFound(crErr) { + return nil + } + // All other errors are unexpected here. + if crErr != nil { + return crErr + } + // If we still find the CR we will return an error. + return errors.New("found Eventing CR, but wanted the Eventing CR to be deleted") + }) + require.NoError(t, err) +} + +// Test_NoPublisherServiceAccountExists tests if the publisher-proxy ServiceAccount was deleted. +func Test_NoPublisherServiceAccountExists(t *testing.T) { + t.Parallel() + ctx := context.TODO() + eventingCR := EventingCR(eventingv1alpha1.BackendType(testConfigs.BackendType)) + err := Retry(attempts, interval, func() error { + _, getErr := clientSet.CoreV1().ServiceAccounts(NamespaceName).Get(ctx, + eventing.GetPublisherServiceAccountName(*eventingCR), metav1.GetOptions{}) + if getErr == nil { + return errors.New("PublisherServiceAccount should have been deleted") + } + if !k8serrors.IsNotFound(getErr) { + return getErr + } + return nil + }) + require.NoError(t, err) +} + +// Enable this test once https://github.com/kyma-project/eventing-manager/issues/34 is done! +//// Test_NoPublisherClusterRoleExists tests if the publisher-proxy ClusterRole was deleted. +//func Test_NoPublisherClusterRoleExists(t *testing.T) { +// t.Parallel() +// ctx := context.TODO() +// eventingCR := EventingCR(eventingv1alpha1.BackendType(testConfigs.BackendType)) +// err := Retry(attempts, interval, func() error { +// _, getErr := clientSet.RbacV1().ClusterRoles().Get(ctx, +// eventing.GetPublisherClusterRoleName(*eventingCR), metav1.GetOptions{}) +// if getErr == nil { +// return errors.New("PublisherClusterRole should have been deleted") +// } +// if !k8serrors.IsNotFound(getErr) { +// return getErr +// } +// return nil +// }) +// require.NoError(t, err) +//} + +// Enable this test once https://github.com/kyma-project/eventing-manager/issues/34 is done! +//// Test_NoPublisherClusterRoleBindingExists tests if the publisher-proxy ClusterRoleBinding was deleted. +//func Test_NoPublisherClusterRoleBindingExists(t *testing.T) { +// t.Parallel() +// ctx := context.TODO() +// eventingCR := EventingCR(eventingv1alpha1.BackendType(testConfigs.BackendType)) +// err := Retry(attempts, interval, func() error { +// _, getErr := clientSet.RbacV1().ClusterRoleBindings().Get(ctx, +// eventing.GetPublisherClusterRoleBindingName(*eventingCR), metav1.GetOptions{}) +// if getErr == nil { +// return errors.New("PublisherClusterRoleBinding should have been deleted") +// } +// if !k8serrors.IsNotFound(getErr) { +// return getErr +// } +// return nil +// }) +// require.NoError(t, err) +//} + +// Test_NoPublisherServicesExists tests if the publisher-proxy Services was deleted. +func Test_NoPublisherServicesExists(t *testing.T) { + t.Parallel() + ctx := context.TODO() + eventingCR := EventingCR(eventingv1alpha1.BackendType(testConfigs.BackendType)) + err := Retry(attempts, interval, func() error { + // check service to expose event publishing endpoint, was deleted. + _, getErr := clientSet.CoreV1().Services(NamespaceName).Get(ctx, + eventing.GetPublisherPublishServiceName(*eventingCR), metav1.GetOptions{}) + if getErr == nil { + return errors.New("Publisher PublishService should have been deleted") + } + if !k8serrors.IsNotFound(getErr) { + return getErr + } + + // check service to expose metrics endpoint, was deleted. + _, getErr = clientSet.CoreV1().Services(NamespaceName).Get(ctx, + eventing.GetPublisherMetricsServiceName(*eventingCR), metav1.GetOptions{}) + if getErr == nil { + return errors.New("Publisher MetricsService should have been deleted") + } + if !k8serrors.IsNotFound(getErr) { + return getErr + } + + // check service to expose health endpoint, was deleted. + _, getErr = clientSet.CoreV1().Services(NamespaceName).Get(ctx, + eventing.GetPublisherHealthServiceName(*eventingCR), metav1.GetOptions{}) + if getErr == nil { + return errors.New("Publisher HealthService should have been deleted") + } + if !k8serrors.IsNotFound(getErr) { + return getErr + } + return nil + }) + require.NoError(t, err) +} + +// Test_PublisherHPA tests the publisher-proxy HorizontalPodAutoscaler was deleted. +func Test_NoPublisherHPAExists(t *testing.T) { + t.Parallel() + ctx := context.TODO() + eventingCR := EventingCR(eventingv1alpha1.BackendType(testConfigs.BackendType)) + err := Retry(attempts, interval, func() error { + _, getErr := clientSet.AutoscalingV2().HorizontalPodAutoscalers(NamespaceName).Get(ctx, + eventing.GetPublisherDeploymentName(*eventingCR), metav1.GetOptions{}) + if !k8serrors.IsNotFound(getErr) { + return getErr + } + return nil + }) + require.NoError(t, err) +} + +// Test_PublisherProxyDeployment checks the publisher-proxy deployment was deleted. +func Test_NoPublisherProxyDeploymentExists(t *testing.T) { + t.Parallel() + ctx := context.TODO() + eventingCR := EventingCR(eventingv1alpha1.BackendType(testConfigs.BackendType)) + err := Retry(attempts, interval, func() error { + _, getErr := clientSet.AppsV1().Deployments(NamespaceName).Get(ctx, + eventing.GetPublisherDeploymentName(*eventingCR), metav1.GetOptions{}) + if !k8serrors.IsNotFound(getErr) { + return getErr + } + return nil + }) + require.NoError(t, err) +} + +func getEventingCR(ctx context.Context, name, namespace string) (*eventingv1alpha1.Eventing, error) { + var eventingCR eventingv1alpha1.Eventing + err := k8sClient.Get(ctx, k8stypes.NamespacedName{ + Name: name, + Namespace: namespace, + }, &eventingCR) + return &eventingCR, err +} diff --git a/hack/e2e/common/fixtures/fixtures.go b/hack/e2e/common/fixtures/fixtures.go new file mode 100644 index 00000000..4ef33b34 --- /dev/null +++ b/hack/e2e/common/fixtures/fixtures.go @@ -0,0 +1,121 @@ +package fixtures + +import ( + "fmt" + "strings" + + eventingv1alpha1 "github.com/kyma-project/eventing-manager/api/v1alpha1" + corev1 "k8s.io/api/core/v1" + "k8s.io/apimachinery/pkg/api/resource" + metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" +) + +const ( + NamespaceName = "kyma-system" + ManagerDeploymentName = "eventing-manager" + CRName = "eventing" + ManagerContainerName = "manager" + PublisherContainerName = "eventing-publisher-proxy" + WebhookServerCertSecretName = "eventing-manager-webhook-server-cert" //nolint:gosec // This is used for test purposes only. + WebhookServerCertJobName = "eventing-manager-cert-handler" + EventMeshSecretNamespace = "kyma-system" + EventMeshSecretName = "eventing-backend" +) + +func EventingCR(backendType eventingv1alpha1.BackendType) *eventingv1alpha1.Eventing { + if backendType == eventingv1alpha1.EventMeshBackendType { + return EventingEventMeshCR() + } + return EventingNATSCR() +} + +func EventingNATSCR() *eventingv1alpha1.Eventing { + return &eventingv1alpha1.Eventing{ + TypeMeta: metav1.TypeMeta{ + Kind: "Eventing", + APIVersion: "operator.kyma-project.io/v1alpha1", + }, + ObjectMeta: metav1.ObjectMeta{ + Name: CRName, + Namespace: NamespaceName, + }, + Spec: eventingv1alpha1.EventingSpec{ + Backend: eventingv1alpha1.Backend{ + Type: "NATS", + Config: eventingv1alpha1.BackendConfig{ + NATSStreamStorageType: "File", + NATSStreamReplicas: 3, + NATSStreamMaxSize: resource.MustParse("700m"), + NATSMaxMsgsPerTopic: 1000000, + }, + }, + Publisher: PublisherSpec(), + }, + } +} + +func EventingEventMeshCR() *eventingv1alpha1.Eventing { + return &eventingv1alpha1.Eventing{ + TypeMeta: metav1.TypeMeta{ + Kind: "Eventing", + APIVersion: "operator.kyma-project.io/v1alpha1", + }, + ObjectMeta: metav1.ObjectMeta{ + Name: CRName, + Namespace: NamespaceName, + }, + Spec: eventingv1alpha1.EventingSpec{ + Backend: eventingv1alpha1.Backend{ + Type: "EventMesh", + Config: eventingv1alpha1.BackendConfig{ + EventMeshSecret: fmt.Sprintf("%s/%s", EventMeshSecretNamespace, EventMeshSecretName), + }, + }, + Publisher: PublisherSpec(), + }, + } +} + +func PublisherSpec() eventingv1alpha1.Publisher { + return eventingv1alpha1.Publisher{ + Replicas: eventingv1alpha1.Replicas{ + Min: 2, + Max: 2, + }, + Resources: corev1.ResourceRequirements{ + Limits: corev1.ResourceList{ + "cpu": resource.MustParse("300m"), + "memory": resource.MustParse("312Mi"), + }, + Requests: corev1.ResourceList{ + "cpu": resource.MustParse("100m"), + "memory": resource.MustParse("156Mi"), + }, + }, + } +} + +func Namespace() *corev1.Namespace { + return &corev1.Namespace{ + ObjectMeta: metav1.ObjectMeta{ + Name: NamespaceName, + }, + } +} + +func FindContainerInPod(pod corev1.Pod, name string) *corev1.Container { + for _, container := range pod.Spec.Containers { + if container.Name == name { + return &container + } + } + return nil +} + +func ConvertSelectorLabelsToString(labels map[string]string) string { + var result []string + for k, v := range labels { + result = append(result, fmt.Sprintf("%s=%s", k, v)) + } + return strings.Join(result, ",") +} diff --git a/hack/e2e/common/k8s.go b/hack/e2e/common/k8s.go new file mode 100644 index 00000000..61252f6b --- /dev/null +++ b/hack/e2e/common/k8s.go @@ -0,0 +1,64 @@ +package common + +import ( + "os" + "path/filepath" + + eventingv1alpha1 "github.com/kyma-project/eventing-manager/api/v1alpha1" + + "k8s.io/client-go/kubernetes" + "k8s.io/client-go/kubernetes/scheme" + "k8s.io/client-go/rest" + "k8s.io/client-go/tools/clientcmd" + "sigs.k8s.io/controller-runtime/pkg/client" + + natsv1alpha1 "github.com/kyma-project/nats-manager/api/v1alpha1" +) + +func GetK8sClients() (*kubernetes.Clientset, client.Client, error) { + kubeConfigPath := "" + if _, ok := os.LookupEnv("KUBECONFIG"); ok { + kubeConfigPath = os.Getenv("KUBECONFIG") + } else { + userHomeDir, err := os.UserHomeDir() + if err != nil { + return nil, nil, err + } + kubeConfigPath = filepath.Join(userHomeDir, ".kube", "config") + } + + var kubeConfig *rest.Config + kubeConfig, err := clientcmd.BuildConfigFromFlags("", kubeConfigPath) + if err != nil { + return nil, nil, err + } + + // Set up the clientSet that is used to access regular K8s objects. + var clientSet *kubernetes.Clientset + clientSet, err = kubernetes.NewForConfig(kubeConfig) + if err != nil { + return nil, nil, err + } + + // We need to add the NATS CRD to the scheme, so we can create a client that can access NATS objects. + err = natsv1alpha1.AddToScheme(scheme.Scheme) + if err != nil { + return nil, nil, err + } + + // We need to add the Eventing CRD to the scheme, so we can create a client that can access Eventing objects. + err = eventingv1alpha1.AddToScheme(scheme.Scheme) + if err != nil { + return nil, nil, err + } + + // Set up the k8s client, so we can access NATS CR-objects. + // +kubebuilder:scaffold:scheme + var k8sClient client.Client + k8sClient, err = client.New(kubeConfig, client.Options{Scheme: scheme.Scheme}) + if err != nil { + return nil, nil, err + } + + return clientSet, k8sClient, nil +} diff --git a/hack/e2e/common/logger.go b/hack/e2e/common/logger.go new file mode 100644 index 00000000..56422c7a --- /dev/null +++ b/hack/e2e/common/logger.go @@ -0,0 +1,36 @@ +package common + +import ( + "os" + + "go.uber.org/zap" + "go.uber.org/zap/zapcore" +) + +func SetupLogger() (*zap.Logger, error) { + logLevel := os.Getenv("E2E_LOG_LEVEL") + + var level zapcore.Level + switch logLevel { + case "debug": + level = zap.DebugLevel + case "info": + level = zap.InfoLevel + case "warn": + level = zap.WarnLevel + case "error": + level = zap.ErrorLevel + default: + level = zap.DebugLevel + } + + config := zap.Config{ + Level: zap.NewAtomicLevelAt(level), + Development: false, + Encoding: "json", + EncoderConfig: zap.NewProductionEncoderConfig(), + OutputPaths: []string{"stdout"}, + ErrorOutputPaths: []string{"stderr"}, + } + return config.Build() +} diff --git a/hack/e2e/common/retry.go b/hack/e2e/common/retry.go new file mode 100644 index 00000000..ea831b98 --- /dev/null +++ b/hack/e2e/common/retry.go @@ -0,0 +1,38 @@ +package common + +import ( + "time" +) + +func Retry(attempts int, interval time.Duration, fn func() error) error { + ticker := time.NewTicker(interval) + defer ticker.Stop() + var err error + for { //nolint:gosimple//There is no range here. + select { + case <-ticker.C: + attempts-- + err = fn() + if err == nil || attempts == 0 { + return err + } + } + } +} + +func RetryGet[T any](attempts int, interval time.Duration, fn func() (*T, error)) (*T, error) { + ticker := time.NewTicker(interval) + defer ticker.Stop() + var err error + var obj *T + for { //nolint:gosimple//There is no range here. + select { + case <-ticker.C: + attempts-- + obj, err = fn() + if err == nil || attempts == 0 { + return obj, err + } + } + } +} diff --git a/hack/e2e/env/env.go b/hack/e2e/env/env.go new file mode 100644 index 00000000..a3ae8301 --- /dev/null +++ b/hack/e2e/env/env.go @@ -0,0 +1,28 @@ +package env + +import ( + "github.com/kelseyhightower/envconfig" + eventingv1alpha1 "github.com/kyma-project/eventing-manager/api/v1alpha1" +) + +// E2EConfig represents the environment config for the end-to-end tests for eventing-manager. +type E2EConfig struct { + BackendType string `envconfig:"BACKEND_TYPE" default:"NATS"` // NATS or EventMesh + ManagerImage string `envconfig:"MANAGER_IMAGE" default:""` +} + +func (cfg E2EConfig) IsNATSBackend() bool { + return eventingv1alpha1.BackendType(cfg.BackendType) == eventingv1alpha1.NatsBackendType +} + +func (cfg E2EConfig) IsEventMeshBackend() bool { + return eventingv1alpha1.BackendType(cfg.BackendType) == eventingv1alpha1.EventMeshBackendType +} + +func GetE2EConfig() (*E2EConfig, error) { + cfg := E2EConfig{} + if err := envconfig.Process("", &cfg); err != nil { + return nil, err + } + return &cfg, nil +} diff --git a/hack/e2e/setup/setup_test.go b/hack/e2e/setup/setup_test.go new file mode 100644 index 00000000..616807b0 --- /dev/null +++ b/hack/e2e/setup/setup_test.go @@ -0,0 +1,475 @@ +//go:build e2e +// +build e2e + +// Package setup_test is part of the end-to-end-tests. This package contains tests that evaluate the creation of a +// eventing CR and the creation of all correlated Kubernetes resources. +// To run the tests a Kubernetes cluster and Eventing-CR need to be available and configured. For this reason, the tests +// are seperated via the `e2e` buildtags. For more information please consult the `readme.md`. +package setup_test + +import ( + "context" + "fmt" + "os" + "reflect" + "testing" + "time" + + eventingv1alpha1 "github.com/kyma-project/eventing-manager/api/v1alpha1" + "github.com/kyma-project/eventing-manager/hack/e2e/env" + "github.com/kyma-project/eventing-manager/pkg/eventing" + "github.com/pkg/errors" + + "github.com/stretchr/testify/require" + "go.uber.org/zap" + appsv1 "k8s.io/api/apps/v1" + k8serrors "k8s.io/apimachinery/pkg/api/errors" + metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" + k8stypes "k8s.io/apimachinery/pkg/types" + "k8s.io/client-go/kubernetes" + "sigs.k8s.io/controller-runtime/pkg/client" + + . "github.com/kyma-project/eventing-manager/hack/e2e/common" + . "github.com/kyma-project/eventing-manager/hack/e2e/common/fixtures" +) + +// Constants for retries. +const ( + interval = 2 * time.Second + attempts = 60 +) + +// clientSet is what is used to access K8s build-in resources like Pods, Namespaces and so on. +var clientSet *kubernetes.Clientset //nolint:gochecknoglobals // This will only be accessible in e2e tests. + +// k8sClient is what is used to access the Eventing CR. +var k8sClient client.Client //nolint:gochecknoglobals // This will only be accessible in e2e tests. + +var logger *zap.Logger + +var testConfigs *env.E2EConfig + +// TestMain runs before all the other test functions. It sets up all the resources that are shared between the different +// test functions. It will then run the tests and finally shuts everything down. +func TestMain(m *testing.M) { + var err error + logger, err = SetupLogger() + if err != nil { + logger.Fatal(err.Error()) + } + + testConfigs, err = env.GetE2EConfig() + if err != nil { + logger.Fatal(err.Error()) + } + logger.Info(fmt.Sprintf("##### NOTE: Tests will run w.r.t. backend: %s", testConfigs.BackendType)) + + clientSet, k8sClient, err = GetK8sClients() + if err != nil { + logger.Fatal(err.Error()) + } + + ctx := context.TODO() + // Create the Namespace used for testing. + err = Retry(attempts, interval, func() error { + // It's fine if the Namespace already exists. + return client.IgnoreAlreadyExists(k8sClient.Create(ctx, Namespace())) + }) + if err != nil { + logger.Fatal(err.Error()) + } + + // Wait for eventing-manager deployment to get ready. + if testConfigs.ManagerImage == "" { + logger.Warn( + "ENV `MANAGER_IMAGE` is not set. Test will not verify if the " + + "manager deployment image is correct or not.", + ) + } + if err := waitForEventingManagerDeploymentReady(testConfigs.ManagerImage); err != nil { + logger.Fatal(err.Error()) + } + + // Create the Eventing CR used for testing. + err = Retry(attempts, interval, func() error { + eventingCR := EventingCR(eventingv1alpha1.BackendType(testConfigs.BackendType)) + errEvnt := k8sClient.Create(ctx, eventingCR) + if k8serrors.IsAlreadyExists(errEvnt) { + gotEventingCR, getErr := getEventingCR(ctx, eventingCR.Name, eventingCR.Namespace) + if getErr != nil { + return err + } + + // If Backend type is changed then update the CR. + if gotEventingCR.Spec.Backend.Type != eventingCR.Spec.Backend.Type { + eventingCR.ObjectMeta = gotEventingCR.ObjectMeta + if errEvnt = k8sClient.Update(ctx, eventingCR); getErr != nil { + return err + } + } else { + logger.Warn( + "error while creating Eventing CR, resource already exist; test will continue with existing CR", + ) + } + return nil + } + return errEvnt + }) + if err != nil { + logger.Fatal(err.Error()) + } + + // wait for an interval for reconciliation to update status. + time.Sleep(interval) + + // Wait for Eventing CR to get ready. + if err := waitForEventingCRReady(); err != nil { + logger.Fatal(err.Error()) + } + + // Run the tests and exit. + code := m.Run() + os.Exit(code) +} + +// Test_WebhookServerCertSecret tests if the Secret exists. +func Test_WebhookServerCertSecret(t *testing.T) { + t.Parallel() + ctx := context.TODO() + err := Retry(attempts, interval, func() error { + _, getErr := clientSet.CoreV1().Secrets(NamespaceName).Get(ctx, WebhookServerCertSecretName, metav1.GetOptions{}) + if getErr != nil { + return getErr + } + return nil + }) + require.NoError(t, err) +} + +// Test_WebhookServerCertJob tests if the Job exists. +func Test_WebhookServerCertJob(t *testing.T) { + t.Parallel() + ctx := context.TODO() + err := Retry(attempts, interval, func() error { + _, getErr := clientSet.BatchV1().Jobs(NamespaceName).Get(ctx, WebhookServerCertJobName, metav1.GetOptions{}) + if getErr != nil { + return getErr + } + return nil + }) + require.NoError(t, err) +} + +// Test_WebhookServerCertCronJob tests if the CronJob exists. +func Test_WebhookServerCertCronJob(t *testing.T) { + t.Parallel() + ctx := context.TODO() + err := Retry(attempts, interval, func() error { + _, getErr := clientSet.BatchV1().CronJobs(NamespaceName).Get(ctx, WebhookServerCertJobName, metav1.GetOptions{}) + if getErr != nil { + return getErr + } + return nil + }) + require.NoError(t, err) +} + +// Test_PublisherServiceAccount tests if the publisher-proxy ServiceAccount exists. +func Test_PublisherServiceAccount(t *testing.T) { + t.Parallel() + ctx := context.TODO() + eventingCR := EventingCR(eventingv1alpha1.BackendType(testConfigs.BackendType)) + err := Retry(attempts, interval, func() error { + _, getErr := clientSet.CoreV1().ServiceAccounts(NamespaceName).Get(ctx, + eventing.GetPublisherServiceAccountName(*eventingCR), metav1.GetOptions{}) + if getErr != nil { + return getErr + } + return nil + }) + require.NoError(t, err) +} + +// Test_PublisherClusterRole tests if the publisher-proxy ClusterRole exists. +func Test_PublisherClusterRole(t *testing.T) { + t.Parallel() + ctx := context.TODO() + eventingCR := EventingCR(eventingv1alpha1.BackendType(testConfigs.BackendType)) + err := Retry(attempts, interval, func() error { + _, getErr := clientSet.RbacV1().ClusterRoles().Get(ctx, + eventing.GetPublisherClusterRoleName(*eventingCR), metav1.GetOptions{}) + if getErr != nil { + return getErr + } + return nil + }) + require.NoError(t, err) +} + +// Test_PublisherClusterRoleBinding tests if the publisher-proxy ClusterRoleBinding exists. +func Test_PublisherClusterRoleBinding(t *testing.T) { + t.Parallel() + ctx := context.TODO() + eventingCR := EventingCR(eventingv1alpha1.BackendType(testConfigs.BackendType)) + err := Retry(attempts, interval, func() error { + _, getErr := clientSet.RbacV1().ClusterRoleBindings().Get(ctx, + eventing.GetPublisherClusterRoleBindingName(*eventingCR), metav1.GetOptions{}) + if getErr != nil { + return getErr + } + return nil + }) + require.NoError(t, err) +} + +// Test_PublisherServices tests if the publisher-proxy Services exists. +func Test_PublisherServices(t *testing.T) { + t.Parallel() + ctx := context.TODO() + eventingCR := EventingCR(eventingv1alpha1.BackendType(testConfigs.BackendType)) + err := Retry(attempts, interval, func() error { + // check service to expose event publishing endpoint. + _, getErr := clientSet.CoreV1().Services(NamespaceName).Get(ctx, + eventing.GetPublisherPublishServiceName(*eventingCR), metav1.GetOptions{}) + if getErr != nil { + return errors.Wrap(getErr, "failed to ensure existence of publish service") + } + + // check service to expose metrics endpoint. + _, getErr = clientSet.CoreV1().Services(NamespaceName).Get(ctx, + eventing.GetPublisherMetricsServiceName(*eventingCR), metav1.GetOptions{}) + if getErr != nil { + return errors.Wrap(getErr, "failed to ensure existence of metrics service") + } + + // check service to expose health endpoint. + _, getErr = clientSet.CoreV1().Services(NamespaceName).Get(ctx, + eventing.GetPublisherHealthServiceName(*eventingCR), metav1.GetOptions{}) + if getErr != nil { + return errors.Wrap(getErr, "failed to ensure existence of health service") + } + + return nil + }) + require.NoError(t, err) +} + +// Test_PublisherHPA tests the publisher-proxy HorizontalPodAutoscaler. +func Test_PublisherHPA(t *testing.T) { + t.Parallel() + ctx := context.TODO() + eventingCR := EventingCR(eventingv1alpha1.BackendType(testConfigs.BackendType)) + err := Retry(attempts, interval, func() error { + gotHPA, getErr := clientSet.AutoscalingV2().HorizontalPodAutoscalers(NamespaceName).Get(ctx, + eventing.GetPublisherDeploymentName(*eventingCR), metav1.GetOptions{}) + if getErr != nil { + return getErr + } + + // check if the MinReplicas is correct. + if int(*gotHPA.Spec.MinReplicas) != eventingCR.Spec.Min { + return fmt.Errorf("HPA MinReplicas do not match. Want: %v, Got: %v", + eventingCR.Spec.Min, int(*gotHPA.Spec.MinReplicas)) + } + + // check if the MaxReplicas is correct. + if int(gotHPA.Spec.MaxReplicas) != eventingCR.Spec.Max { + return fmt.Errorf("HPA MinReplicas do not match. Want: %v, Got: %v", + eventingCR.Spec.Max, gotHPA.Spec.MaxReplicas) + } + + return nil + }) + require.NoError(t, err) +} + +// Test_PublisherProxyDeployment checks the publisher-proxy deployment. +func Test_PublisherProxyDeployment(t *testing.T) { + t.Parallel() + ctx := context.TODO() + eventingCR := EventingCR(eventingv1alpha1.BackendType(testConfigs.BackendType)) + err := Retry(attempts, interval, func() error { + gotDeployment, getErr := getDeployment(ctx, eventing.GetPublisherDeploymentName(*eventingCR), NamespaceName) + if getErr != nil { + return getErr + } + + // check if the deployment is ready. + if gotDeployment.Status.Replicas != gotDeployment.Status.UpdatedReplicas || + gotDeployment.Status.Replicas != gotDeployment.Status.ReadyReplicas { + err := fmt.Errorf("waiting for publisher-proxy deployment to get ready") + logger.Debug(err.Error()) + return err + } + + return nil + }) + require.NoError(t, err) +} + +// Test_PublisherProxyPods checks the publisher-proxy pods. +// - checks if number of pods are between spec min and max. +// - checks if the ENV `BACKEND` is correctly defined for active backend. +// - checks if container resources are as defined in eventing CR spec. +func Test_PublisherProxyPods(t *testing.T) { + t.Parallel() + + ctx := context.TODO() + eventingCR := EventingCR(eventingv1alpha1.BackendType(testConfigs.BackendType)) + // RetryGet the Pods and test them. + err := Retry(attempts, interval, func() error { + // get publisher deployment + gotDeployment, getErr := getDeployment(ctx, eventing.GetPublisherDeploymentName(*eventingCR), NamespaceName) + if getErr != nil { + return getErr + } + + // RetryGet the Pods via labels. + pods, err := clientSet.CoreV1().Pods(NamespaceName).List(ctx, metav1.ListOptions{ + LabelSelector: ConvertSelectorLabelsToString(gotDeployment.Spec.Selector.MatchLabels)}) + if err != nil { + return err + } + + // check number of replicas of Publisher pods. + if len(pods.Items) < eventingCR.Spec.Publisher.Min || len(pods.Items) > eventingCR.Spec.Publisher.Max { + return fmt.Errorf( + "the number of replicas for Publisher pods do not match with Eventing CR spec. "+ + "Wanted replicas to be between: %v and %v, but got %v", + eventingCR.Spec.Publisher.Min, + eventingCR.Spec.Publisher.Max, + len(pods.Items), + ) + } + + // Go through all Pods, check its spec. It should be same as defined in Eventing CR + for _, pod := range pods.Items { + // find the container. + container := FindContainerInPod(pod, PublisherContainerName) + if container == nil { + return fmt.Errorf("Container: %v not found in publisher pod.", PublisherContainerName) + } + // compare the Resources with what is defined in the Eventing CR. + if !reflect.DeepEqual(eventingCR.Spec.Publisher.Resources, container.Resources) { + return fmt.Errorf( + "error when checking pod %s resources:\n\twanted: %s\n\tgot: %s", + pod.GetName(), + eventingCR.Spec.Publisher.Resources.String(), + container.Resources.String(), + ) + } + + // check if the ENV `BACKEND` is defined correctly. + wantBackendENVValue := "nats" + if eventingv1alpha1.BackendType(testConfigs.BackendType) == eventingv1alpha1.EventMeshBackendType { + wantBackendENVValue = "beb" + } + + // get value of ENV `BACKEND` + gotBackendENVValue := "" + for _, envVar := range container.Env { + if envVar.Name == "BACKEND" { + gotBackendENVValue = envVar.Value + } + } + // compare value + if wantBackendENVValue != gotBackendENVValue { + return fmt.Errorf( + `error when checking ENV [BACKEND], wanted: %s, got: %s`, + wantBackendENVValue, + gotBackendENVValue, + ) + } + } + + // Everything is fine. + return nil + }) + require.NoError(t, err) +} + +func getEventingCR(ctx context.Context, name, namespace string) (*eventingv1alpha1.Eventing, error) { + var eventingCR eventingv1alpha1.Eventing + err := k8sClient.Get(ctx, k8stypes.NamespacedName{ + Name: name, + Namespace: namespace, + }, &eventingCR) + return &eventingCR, err +} + +func getDeployment(ctx context.Context, name, namespace string) (*appsv1.Deployment, error) { + return clientSet.AppsV1().Deployments(namespace).Get(ctx, name, metav1.GetOptions{}) +} + +// Wait for Eventing CR to get ready. +func waitForEventingCRReady() error { + // RetryGet the Eventing CR and test status. + return Retry(attempts, interval, func() error { + logger.Debug(fmt.Sprintf("waiting for Eventing CR to get ready. "+ + "CR name: %s, namespace: %s", CRName, NamespaceName)) + + ctx := context.TODO() + // Get the Eventing CR from the cluster. + gotEventingCR, err := RetryGet(attempts, interval, func() (*eventingv1alpha1.Eventing, error) { + return getEventingCR(ctx, CRName, NamespaceName) + }) + if err != nil { + return err + } + + if gotEventingCR.Spec.Backend.Type != gotEventingCR.Status.ActiveBackend { + err := fmt.Errorf("waiting for Eventing CR to switch backend") + logger.Debug(err.Error()) + return err + } + + if gotEventingCR.Status.State != eventingv1alpha1.StateReady { + err := fmt.Errorf("waiting for Eventing CR to get ready state") + logger.Debug(err.Error()) + return err + } + + // Everything is fine. + logger.Debug(fmt.Sprintf("Eventing CR is ready. "+ + "CR name: %s, namespace: %s", CRName, NamespaceName)) + return nil + }) +} + +// Wait for eventing-manager deployment to get ready with correct image. +func waitForEventingManagerDeploymentReady(image string) error { + // RetryGet the Eventing Manager and test status. + return Retry(attempts, interval, func() error { + logger.Debug(fmt.Sprintf("waiting for eventing-manager deployment to get ready with image: %s", image)) + ctx := context.TODO() + // Get the eventing-manager deployment from the cluster. + gotDeployment, err := RetryGet(attempts, interval, func() (*appsv1.Deployment, error) { + return getDeployment(ctx, ManagerDeploymentName, NamespaceName) + }) + if err != nil { + return err + } + + // if image is provided, then check if the deployment has correct image. + if image != "" && gotDeployment.Spec.Template.Spec.Containers[0].Image != image { + err := fmt.Errorf("expected eventing-manager image to be: %s, but found: %s", image, + gotDeployment.Spec.Template.Spec.Containers[0].Image, + ) + logger.Debug(err.Error()) + return err + } + + // check if the deployment is ready. + if *gotDeployment.Spec.Replicas != gotDeployment.Status.UpdatedReplicas || + *gotDeployment.Spec.Replicas != gotDeployment.Status.ReadyReplicas || + *gotDeployment.Spec.Replicas != gotDeployment.Status.AvailableReplicas { + err := fmt.Errorf("waiting for eventing-manager deployment to get ready") + logger.Debug(err.Error()) + return err + } + + // Everything is fine. + logger.Debug(fmt.Sprintf("eventing-manager deployment is ready with image: %s", image)) + return nil + }) +}