From 6ec02d61b8079425447d15c7799855feb1013534 Mon Sep 17 00:00:00 2001 From: Jason Parraga Date: Wed, 20 Nov 2024 22:50:05 -0800 Subject: [PATCH 1/3] Add liveness and readiness probes Signed-off-by: Jason Parraga --- .../install/armadaserver_controller.go | 27 +++ .../install/binoculars_controller.go | 29 +++ internal/controller/install/common_helpers.go | 58 ++++- .../controller/install/executor_controller.go | 4 + .../controller/install/lookout_controller.go | 31 ++- .../install/lookout_controller_test.go | 197 +++++++++++++++++ .../install/scheduler_controller.go | 32 ++- .../install/scheduler_controller_test.go | 202 ++++++++++++++++++ 8 files changed, 557 insertions(+), 23 deletions(-) diff --git a/internal/controller/install/armadaserver_controller.go b/internal/controller/install/armadaserver_controller.go index c622228..abd2e05 100644 --- a/internal/controller/install/armadaserver_controller.go +++ b/internal/controller/install/armadaserver_controller.go @@ -449,6 +449,13 @@ func createArmadaServerDeployment( volumeMounts := createVolumeMounts(GetConfigFilename(as.Name), as.Spec.AdditionalVolumeMounts) volumeMounts = append(volumeMounts, createPulsarVolumeMounts(pulsarConfig)...) + asConfig, err := extractArmadaServerConfig(as.Spec.ApplicationConfig) + if err != nil { + return nil, err + } + + readinessProbe, livenessProbe := CreateProbesWithScheme(GetServerScheme(asConfig.Grpc.Tls)) + deployment := appsv1.Deployment{ ObjectMeta: metav1.ObjectMeta{ Name: as.Name, @@ -483,6 +490,8 @@ func createArmadaServerDeployment( Env: env, VolumeMounts: volumeMounts, SecurityContext: as.Spec.SecurityContext, + ReadinessProbe: readinessProbe, + LivenessProbe: livenessProbe, }}, Volumes: volumes, }, @@ -678,3 +687,21 @@ func createServerPrometheusRule(server *installv1alpha1.ArmadaServer) *monitorin }, } } + +type ArmadaServerConfig struct { + Grpc GrpcConfig +} + +// extractArmadaServerConfig will unmarshal the appconfig and return the AramadaServerConfig portion +func extractArmadaServerConfig(config runtime.RawExtension) (ArmadaServerConfig, error) { + appConfig, err := builders.ConvertRawExtensionToYaml(config) + if err != nil { + return ArmadaServerConfig{}, err + } + var asConfig ArmadaServerConfig + err = yaml.Unmarshal([]byte(appConfig), &asConfig) + if err != nil { + return ArmadaServerConfig{}, err + } + return asConfig, err +} diff --git a/internal/controller/install/binoculars_controller.go b/internal/controller/install/binoculars_controller.go index d7c80a0..5917347 100644 --- a/internal/controller/install/binoculars_controller.go +++ b/internal/controller/install/binoculars_controller.go @@ -20,6 +20,8 @@ import ( "context" "time" + "sigs.k8s.io/yaml" + "github.com/pkg/errors" installv1alpha1 "github.com/armadaproject/armada-operator/api/install/v1alpha1" @@ -227,6 +229,13 @@ func createBinocularsDeployment( env := createEnv(binoculars.Spec.Environment) volumes := createVolumes(binoculars.Name, binoculars.Spec.AdditionalVolumes) volumeMounts := createVolumeMounts(GetConfigFilename(secret.Name), binoculars.Spec.AdditionalVolumeMounts) + + binocularsConfig, err := extractBinocularsConfig(binoculars.Spec.ApplicationConfig) + if err != nil { + return nil, err + } + readinessProbe, livenessProbe := CreateProbesWithScheme(GetServerScheme(binocularsConfig.Grpc.Tls)) + containers := []corev1.Container{{ Name: "binoculars", ImagePullPolicy: corev1.PullIfNotPresent, @@ -236,6 +245,8 @@ func createBinocularsDeployment( Env: env, VolumeMounts: volumeMounts, SecurityContext: binoculars.Spec.SecurityContext, + ReadinessProbe: readinessProbe, + LivenessProbe: livenessProbe, }} deployment := appsv1.Deployment{ ObjectMeta: metav1.ObjectMeta{Name: binoculars.Name, Namespace: binoculars.Namespace, Labels: AllLabels(binoculars.Name, binoculars.Labels)}, @@ -367,3 +378,21 @@ func (r *BinocularsReconciler) SetupWithManager(mgr ctrl.Manager) error { For(&installv1alpha1.Binoculars{}). Complete(r) } + +type BinocularsConfig struct { + Grpc GrpcConfig +} + +// extractBinocularsConfig will unmarshal the appconfig and return the BinocularsConfig portion +func extractBinocularsConfig(config runtime.RawExtension) (BinocularsConfig, error) { + appConfig, err := builders.ConvertRawExtensionToYaml(config) + if err != nil { + return BinocularsConfig{}, err + } + var binocularsConfig BinocularsConfig + err = yaml.Unmarshal([]byte(appConfig), &binocularsConfig) + if err != nil { + return BinocularsConfig{}, err + } + return binocularsConfig, err +} diff --git a/internal/controller/install/common_helpers.go b/internal/controller/install/common_helpers.go index f2bca45..0b61f70 100644 --- a/internal/controller/install/common_helpers.go +++ b/internal/controller/install/common_helpers.go @@ -229,6 +229,14 @@ type PulsarConfig struct { Cacert string } +type GrpcConfig struct { + Tls TlsConfig +} + +type TlsConfig struct { + Enabled bool +} + // ArmadaInit used to initialize pulsar type ArmadaInit struct { Enabled bool @@ -317,7 +325,16 @@ func ExtractPulsarConfig(config runtime.RawExtension) (PulsarConfig, error) { if err != nil { return PulsarConfig{}, err } - return asConfig.Pulsar, nil + return asConfig.Pulsar, err +} + +// GetServerScheme returns the URI scheme for the grpc server +func GetServerScheme(tlsConfig TlsConfig) corev1.URIScheme { + if tlsConfig.Enabled { + return corev1.URISchemeHTTPS + } + + return corev1.URISchemeHTTP } // waitForJob waits for the Job to reach a terminal state (complete or failed). @@ -371,6 +388,36 @@ func createEnv(crdEnv []corev1.EnvVar) []corev1.EnvVar { return envVars } +func CreateProbesWithScheme(scheme corev1.URIScheme) (*corev1.Probe, *corev1.Probe) { + readinessProbe := &corev1.Probe{ + ProbeHandler: corev1.ProbeHandler{ + HTTPGet: &corev1.HTTPGetAction{ + Path: "/health", + Port: intstr.FromString("http"), + Scheme: scheme, + }, + }, + InitialDelaySeconds: 5, + TimeoutSeconds: 5, + FailureThreshold: 2, + } + + livenessProbe := &corev1.Probe{ + ProbeHandler: corev1.ProbeHandler{ + HTTPGet: &corev1.HTTPGetAction{ + Path: "/health", + Port: intstr.FromString("http"), + Scheme: scheme, + }, + }, + InitialDelaySeconds: 10, + TimeoutSeconds: 10, + FailureThreshold: 3, + } + + return readinessProbe, livenessProbe +} + // createVolumes creates the default appconfig Volume and appends the CRD AdditionalVolumes func createVolumes(configVolumeSecretName string, crdVolumes []corev1.Volume) []corev1.Volume { volumes := []corev1.Volume{{ @@ -733,15 +780,6 @@ func newContainerPortsHTTPWithMetrics(config *builders.CommonApplicationConfig) return ports } -// newContainerPortsGRPCWithMetrics creates container ports for grpc and metrics server and optional port for profiling server. -func newContainerPortsGRPCWithMetrics(config *builders.CommonApplicationConfig) []corev1.ContainerPort { - ports := []corev1.ContainerPort{newContainerPortGRPC(config), newContainerPortMetrics(config)} - if config.Profiling.Port > 0 { - ports = append(ports, newContainerPortProfiling(config)) - } - return ports -} - // newContainerPortsMetrics creates container ports for metrics server and optional port for profiling server. func newContainerPortsMetrics(config *builders.CommonApplicationConfig) []corev1.ContainerPort { ports := []corev1.ContainerPort{newContainerPortMetrics(config)} diff --git a/internal/controller/install/executor_controller.go b/internal/controller/install/executor_controller.go index 26d0644..9fac5c0 100644 --- a/internal/controller/install/executor_controller.go +++ b/internal/controller/install/executor_controller.go @@ -250,6 +250,8 @@ func (r *ExecutorReconciler) createDeployment( volumes := createVolumes(executor.Name, executor.Spec.AdditionalVolumes) volumeMounts := createVolumeMounts(GetConfigFilename(executor.Name), executor.Spec.AdditionalVolumeMounts) + readinessProbe, livenessProbe := CreateProbesWithScheme(corev1.URISchemeHTTP) + env := []corev1.EnvVar{ { Name: "SERVICE_ACCOUNT", @@ -278,6 +280,8 @@ func (r *ExecutorReconciler) createDeployment( Env: env, VolumeMounts: volumeMounts, SecurityContext: executor.Spec.SecurityContext, + ReadinessProbe: readinessProbe, + LivenessProbe: livenessProbe, }} deployment := appsv1.Deployment{ ObjectMeta: metav1.ObjectMeta{Name: executor.Name, Namespace: executor.Namespace, Labels: AllLabels(executor.Name, executor.Labels)}, diff --git a/internal/controller/install/lookout_controller.go b/internal/controller/install/lookout_controller.go index 2aecb14..82f7f3b 100644 --- a/internal/controller/install/lookout_controller.go +++ b/internal/controller/install/lookout_controller.go @@ -139,6 +139,7 @@ func (r *LookoutReconciler) Reconcile(ctx context.Context, req ctrl.Request) (ct type LookoutConfig struct { Postgres PostgresConfig + Tls TlsConfig } func generateLookoutInstallComponents( @@ -271,6 +272,13 @@ func createLookoutDeployment(lookout *installv1alpha1.Lookout, serviceAccountNam volumes := createVolumes(lookout.Name, lookout.Spec.AdditionalVolumes) volumeMounts := createVolumeMounts(GetConfigFilename(lookout.Name), lookout.Spec.AdditionalVolumeMounts) + lookoutConfig, err := extractLookoutConfig(lookout.Spec.ApplicationConfig) + if err != nil { + return nil, err + } + + readinessProbe, livenessProbe := CreateProbesWithScheme(GetServerScheme(lookoutConfig.Tls)) + deployment := appsv1.Deployment{ ObjectMeta: metav1.ObjectMeta{Name: lookout.Name, Namespace: lookout.Namespace, Labels: AllLabels(lookout.Name, lookout.Labels)}, Spec: appsv1.DeploymentSpec{ @@ -299,6 +307,8 @@ func createLookoutDeployment(lookout *installv1alpha1.Lookout, serviceAccountNam Env: env, VolumeMounts: volumeMounts, SecurityContext: lookout.Spec.SecurityContext, + ReadinessProbe: readinessProbe, + LivenessProbe: livenessProbe, }}, Volumes: volumes, }, @@ -458,12 +468,7 @@ func createLookoutCronJob(lookout *installv1alpha1.Lookout, serviceAccountName s dbPruningSchedule = *lookout.Spec.DbPruningSchedule } - appConfig, err := builders.ConvertRawExtensionToYaml(lookout.Spec.ApplicationConfig) - if err != nil { - return nil, err - } - var lookoutConfig LookoutConfig - err = yaml.Unmarshal([]byte(appConfig), &lookoutConfig) + lookoutConfig, err := extractLookoutConfig(lookout.Spec.ApplicationConfig) if err != nil { return nil, err } @@ -555,3 +560,17 @@ func (r *LookoutReconciler) SetupWithManager(mgr ctrl.Manager) error { For(&installv1alpha1.Lookout{}). Complete(r) } + +// extractLookoutConfig will unmarshal the appconfig and return the LookoutConfig portion +func extractLookoutConfig(config runtime.RawExtension) (LookoutConfig, error) { + appConfig, err := builders.ConvertRawExtensionToYaml(config) + if err != nil { + return LookoutConfig{}, err + } + var lookoutConfig LookoutConfig + err = yaml.Unmarshal([]byte(appConfig), &lookoutConfig) + if err != nil { + return LookoutConfig{}, err + } + return lookoutConfig, err +} diff --git a/internal/controller/install/lookout_controller_test.go b/internal/controller/install/lookout_controller_test.go index 962a527..91e9593 100644 --- a/internal/controller/install/lookout_controller_test.go +++ b/internal/controller/install/lookout_controller_test.go @@ -5,6 +5,8 @@ import ( "testing" "time" + "k8s.io/apimachinery/pkg/util/intstr" + "github.com/google/go-cmp/cmp" "google.golang.org/protobuf/testing/protocmp" @@ -347,6 +349,201 @@ func TestLookoutReconciler_CreateCronJobErrorDueToApplicationConfig(t *testing.T assert.Equal(t, "yaml: line 1: did not find expected ',' or '}'", err.Error()) } +func TestLookoutReconciler_CreateDeployment(t *testing.T) { + t.Parallel() + + commonConfig := &builders.CommonApplicationConfig{ + HTTPPort: 8080, + GRPCPort: 5051, + MetricsPort: 9000, + Profiling: builders.ProfilingConfig{ + Port: 1337, + }, + } + + lookout := &v1alpha1.Lookout{ + TypeMeta: metav1.TypeMeta{ + Kind: "Lookout", + APIVersion: "install.armadaproject.io/v1alpha1", + }, + ObjectMeta: metav1.ObjectMeta{ + Namespace: "default", + Name: "lookout", + DeletionTimestamp: &metav1.Time{Time: time.Now()}, + Finalizers: []string{operatorFinalizer}, + }, + Spec: v1alpha1.LookoutSpec{ + CommonSpecBase: installv1alpha1.CommonSpecBase{ + Labels: nil, + Image: v1alpha1.Image{ + Repository: "testrepo", + Tag: "1.0.0", + }, + ApplicationConfig: runtime.RawExtension{Raw: []byte(`{}`)}, + Resources: &corev1.ResourceRequirements{}, + }, + Replicas: ptr.To[int32](2), + ClusterIssuer: "test", + Ingress: &v1alpha1.IngressConfig{ + IngressClass: "nginx", + }, + }, + } + + deployment, err := createLookoutDeployment(lookout, "lookout", commonConfig) + assert.NoError(t, err) + + expectedDeployment := &appsv1.Deployment{ + ObjectMeta: metav1.ObjectMeta{ + Name: "lookout", + Namespace: "default", + Labels: map[string]string{ + "app": "lookout", + "release": "lookout", + }, + }, + Spec: appsv1.DeploymentSpec{ + Replicas: ptr.To[int32](2), + Selector: &metav1.LabelSelector{ + MatchLabels: map[string]string{ + "app": "lookout", + }, + }, + Template: corev1.PodTemplateSpec{ + ObjectMeta: metav1.ObjectMeta{ + Name: "lookout", + Namespace: "default", + Labels: map[string]string{ + "app": "lookout", + "release": "lookout", + }, + Annotations: map[string]string{ + "checksum/config": "44136fa355b3678a1146ad16f7e8649e94fb4fc21fe77e8310c060f61caaff8a", + }, + }, + Spec: corev1.PodSpec{ + Affinity: &corev1.Affinity{ + PodAffinity: &corev1.PodAffinity{ + PreferredDuringSchedulingIgnoredDuringExecution: []corev1.WeightedPodAffinityTerm{ + { + Weight: 100, + PodAffinityTerm: corev1.PodAffinityTerm{ + LabelSelector: &metav1.LabelSelector{ + MatchExpressions: []metav1.LabelSelectorRequirement{ + { + Key: "app", + Operator: metav1.LabelSelectorOpIn, + Values: []string{ + "lookout", + }, + }, + }, + }, + TopologyKey: "kubernetes.io/hostname", + }, + }, + }, + }, + }, + Volumes: []corev1.Volume{ + { + Name: "user-config", + VolumeSource: corev1.VolumeSource{ + Secret: &corev1.SecretVolumeSource{ + SecretName: "lookout", + }, + }, + }, + }, + Containers: []corev1.Container{ + { + Args: []string{ + "--config", + "/config/application_config.yaml", + }, + Env: []corev1.EnvVar{ + { + Name: "SERVICE_ACCOUNT", + ValueFrom: &corev1.EnvVarSource{ + FieldRef: &corev1.ObjectFieldSelector{ + FieldPath: "spec.serviceAccountName", + }, + }, + }, + { + Name: "POD_NAMESPACE", + ValueFrom: &corev1.EnvVarSource{ + FieldRef: &corev1.ObjectFieldSelector{ + FieldPath: "metadata.namespace", + }, + }, + }, + }, + Image: "testrepo:1.0.0", + ImagePullPolicy: corev1.PullIfNotPresent, + LivenessProbe: &corev1.Probe{ + ProbeHandler: corev1.ProbeHandler{ + HTTPGet: &corev1.HTTPGetAction{ + Path: "/health", + Port: intstr.FromString("http"), + Scheme: corev1.URISchemeHTTP, + }, + }, + InitialDelaySeconds: 10, + TimeoutSeconds: 10, + FailureThreshold: 3, + }, + Name: "lookout", + Ports: []corev1.ContainerPort{ + { + Name: "http", + ContainerPort: 8080, + Protocol: corev1.ProtocolTCP, + }, + { + Name: "metrics", + ContainerPort: 9000, + Protocol: corev1.ProtocolTCP, + }, + { + Name: "profiling", + ContainerPort: 1337, + Protocol: corev1.ProtocolTCP, + }, + }, + ReadinessProbe: &corev1.Probe{ + ProbeHandler: corev1.ProbeHandler{ + HTTPGet: &corev1.HTTPGetAction{ + Path: "/health", + Port: intstr.FromString("http"), + Scheme: corev1.URISchemeHTTP, + }, + }, + InitialDelaySeconds: 5, + TimeoutSeconds: 5, + FailureThreshold: 2, + }, + VolumeMounts: []corev1.VolumeMount{ + { + Name: "user-config", + ReadOnly: true, + MountPath: appConfigFilepath, + SubPath: "lookout-config.yaml", + }, + }, + }, + }, + ServiceAccountName: "lookout", + }, + }, + }, + } + + if !cmp.Equal(expectedDeployment, deployment, protocmp.Transform()) { + t.Fatalf("deployment is not the same %s", cmp.Diff(expectedDeployment, deployment, protocmp.Transform())) + } +} + func TestLookoutReconciler_CreateCronJob(t *testing.T) { t.Parallel() diff --git a/internal/controller/install/scheduler_controller.go b/internal/controller/install/scheduler_controller.go index b21a6ba..cae19a9 100644 --- a/internal/controller/install/scheduler_controller.go +++ b/internal/controller/install/scheduler_controller.go @@ -141,6 +141,7 @@ func (r *SchedulerReconciler) Reconcile(ctx context.Context, req ctrl.Request) ( } type SchedulerConfig struct { + Grpc GrpcConfig Postgres PostgresConfig } @@ -290,6 +291,12 @@ func newSchedulerDeployment( volumeMounts := createVolumeMounts(GetConfigFilename(scheduler.Name), scheduler.Spec.AdditionalVolumeMounts) volumeMounts = append(volumeMounts, createPulsarVolumeMounts(pulsarConfig)...) + schedulerConfig, err := extractSchedulerConfig(scheduler.Spec.ApplicationConfig) + if err != nil { + return nil, err + } + readinessProbe, livenessProbe := CreateProbesWithScheme(GetServerScheme(schedulerConfig.Grpc.Tls)) + deployment := appsv1.Deployment{ ObjectMeta: metav1.ObjectMeta{Name: scheduler.Name, Namespace: scheduler.Namespace, Labels: AllLabels(scheduler.Name, scheduler.Labels)}, Spec: appsv1.DeploymentSpec{ @@ -314,10 +321,12 @@ func newSchedulerDeployment( ImagePullPolicy: corev1.PullIfNotPresent, Image: ImageString(scheduler.Spec.Image), Args: []string{"run", appConfigFlag, appConfigFilepath}, - Ports: newContainerPortsGRPCWithMetrics(config), + Ports: newContainerPortsAll(config), Env: env, VolumeMounts: volumeMounts, SecurityContext: scheduler.Spec.SecurityContext, + ReadinessProbe: readinessProbe, + LivenessProbe: livenessProbe, }}, Volumes: volumes, }, @@ -367,12 +376,7 @@ func newSchedulerMigrationJob(scheduler *installv1alpha1.Scheduler, serviceAccou volumes := createVolumes(scheduler.Name, scheduler.Spec.AdditionalVolumes) volumeMounts := createVolumeMounts(GetConfigFilename(scheduler.Name), scheduler.Spec.AdditionalVolumeMounts) - appConfig, err := builders.ConvertRawExtensionToYaml(scheduler.Spec.ApplicationConfig) - if err != nil { - return nil, err - } - var schedulerConfig SchedulerConfig - err = yaml.Unmarshal([]byte(appConfig), &schedulerConfig) + schedulerConfig, err := extractSchedulerConfig(scheduler.Spec.ApplicationConfig) if err != nil { return nil, err } @@ -746,3 +750,17 @@ func (r *SchedulerReconciler) SetupWithManager(mgr ctrl.Manager) error { For(&installv1alpha1.Scheduler{}). Complete(r) } + +// extractSchedulerConfig will unmarshal the appconfig and return the SchedulerConfig portion +func extractSchedulerConfig(config runtime.RawExtension) (SchedulerConfig, error) { + appConfig, err := builders.ConvertRawExtensionToYaml(config) + if err != nil { + return SchedulerConfig{}, err + } + var schedulerConfig SchedulerConfig + err = yaml.Unmarshal([]byte(appConfig), &schedulerConfig) + if err != nil { + return SchedulerConfig{}, err + } + return schedulerConfig, err +} diff --git a/internal/controller/install/scheduler_controller_test.go b/internal/controller/install/scheduler_controller_test.go index 5e21513..7849e62 100644 --- a/internal/controller/install/scheduler_controller_test.go +++ b/internal/controller/install/scheduler_controller_test.go @@ -5,6 +5,8 @@ import ( "testing" "time" + "k8s.io/apimachinery/pkg/util/intstr" + "github.com/google/go-cmp/cmp" "google.golang.org/protobuf/testing/protocmp" @@ -486,6 +488,206 @@ func TestSchedulerReconciler_ReconcileMissingResources(t *testing.T) { } } +func TestSchedulerReconciler_CreateDeployment(t *testing.T) { + t.Parallel() + + commonConfig := &builders.CommonApplicationConfig{ + HTTPPort: 8080, + GRPCPort: 5051, + MetricsPort: 9000, + Profiling: builders.ProfilingConfig{ + Port: 1337, + }, + } + + scheduler := &v1alpha1.Scheduler{ + TypeMeta: metav1.TypeMeta{ + Kind: "Scheduler", + APIVersion: "install.armadaproject.io/v1alpha1", + }, + ObjectMeta: metav1.ObjectMeta{Namespace: "default", Name: "scheduler"}, + Spec: v1alpha1.SchedulerSpec{ + Replicas: ptr.To[int32](2), + CommonSpecBase: installv1alpha1.CommonSpecBase{ + Labels: nil, + Image: v1alpha1.Image{ + Repository: "testrepo", + Tag: "1.0.0", + }, + ApplicationConfig: runtime.RawExtension{}, + Prometheus: &installv1alpha1.PrometheusConfig{Enabled: true, ScrapeInterval: &metav1.Duration{Duration: 1 * time.Second}}, + TerminationGracePeriodSeconds: ptr.To(int64(20)), + }, + ClusterIssuer: "test", + HostNames: []string{"localhost"}, + Ingress: &installv1alpha1.IngressConfig{ + IngressClass: "nginx", + Labels: map[string]string{"test": "hello"}, + Annotations: map[string]string{"test": "hello"}, + }, + }, + } + + deployment, err := newSchedulerDeployment(scheduler, "scheduler", commonConfig) + assert.NoError(t, err) + + expectedDeployment := &appsv1.Deployment{ + ObjectMeta: metav1.ObjectMeta{ + Name: "scheduler", + Namespace: "default", + Labels: map[string]string{ + "app": "scheduler", + "release": "scheduler", + }, + }, + Spec: appsv1.DeploymentSpec{ + Replicas: ptr.To[int32](2), + Selector: &metav1.LabelSelector{ + MatchLabels: map[string]string{ + "app": "scheduler", + }, + }, + Template: corev1.PodTemplateSpec{ + ObjectMeta: metav1.ObjectMeta{ + Name: "scheduler", + Namespace: "default", + Labels: map[string]string{ + "app": "scheduler", + "release": "scheduler", + }, + Annotations: map[string]string{ + "checksum/config": "e3b0c44298fc1c149afbf4c8996fb92427ae41e4649b934ca495991b7852b855", + }, + }, + Spec: corev1.PodSpec{ + Affinity: &corev1.Affinity{ + PodAffinity: &corev1.PodAffinity{ + PreferredDuringSchedulingIgnoredDuringExecution: []corev1.WeightedPodAffinityTerm{ + { + Weight: 100, + PodAffinityTerm: corev1.PodAffinityTerm{ + LabelSelector: &metav1.LabelSelector{ + MatchExpressions: []metav1.LabelSelectorRequirement{ + { + Key: "app", + Operator: metav1.LabelSelectorOpIn, + Values: []string{ + "scheduler", + }, + }, + }, + }, + TopologyKey: "kubernetes.io/hostname", + }, + }, + }, + }, + }, + Volumes: []corev1.Volume{ + { + Name: "user-config", + VolumeSource: corev1.VolumeSource{ + Secret: &corev1.SecretVolumeSource{ + SecretName: "scheduler", + }, + }, + }, + }, + Containers: []corev1.Container{ + { + Args: []string{ + "run", + "--config", + "/config/application_config.yaml", + }, + Env: []corev1.EnvVar{ + { + Name: "SERVICE_ACCOUNT", + ValueFrom: &corev1.EnvVarSource{ + FieldRef: &corev1.ObjectFieldSelector{ + FieldPath: "spec.serviceAccountName", + }, + }, + }, + { + Name: "POD_NAMESPACE", + ValueFrom: &corev1.EnvVarSource{ + FieldRef: &corev1.ObjectFieldSelector{ + FieldPath: "metadata.namespace", + }, + }, + }, + }, + Image: "testrepo:1.0.0", + ImagePullPolicy: corev1.PullIfNotPresent, + LivenessProbe: &corev1.Probe{ + ProbeHandler: corev1.ProbeHandler{ + HTTPGet: &corev1.HTTPGetAction{ + Path: "/health", + Port: intstr.FromString("http"), + Scheme: corev1.URISchemeHTTP, + }, + }, + InitialDelaySeconds: 10, + TimeoutSeconds: 10, + FailureThreshold: 3, + }, + Name: "scheduler", + Ports: []corev1.ContainerPort{ + { + Name: "grpc", + ContainerPort: 5051, + Protocol: corev1.ProtocolTCP, + }, + { + Name: "http", + ContainerPort: 8080, + Protocol: corev1.ProtocolTCP, + }, + { + Name: "metrics", + ContainerPort: 9000, + Protocol: corev1.ProtocolTCP, + }, + { + Name: "profiling", + ContainerPort: 1337, + Protocol: corev1.ProtocolTCP, + }, + }, + ReadinessProbe: &corev1.Probe{ + ProbeHandler: corev1.ProbeHandler{ + HTTPGet: &corev1.HTTPGetAction{ + Path: "/health", + Port: intstr.FromString("http"), + Scheme: corev1.URISchemeHTTP, + }, + }, + InitialDelaySeconds: 5, + TimeoutSeconds: 5, + FailureThreshold: 2, + }, + VolumeMounts: []corev1.VolumeMount{ + { + Name: "user-config", + ReadOnly: true, + MountPath: appConfigFilepath, + SubPath: "scheduler-config.yaml", + }, + }, + }, + }, + ServiceAccountName: "scheduler", + }, + }, + }, + } + + if !cmp.Equal(expectedDeployment, deployment, protocmp.Transform()) { + t.Fatalf("deployment is not the same %s", cmp.Diff(expectedDeployment, deployment, protocmp.Transform())) + } +} + func TestSchedulerReconciler_createSchedulerCronJob(t *testing.T) { t.Parallel() From d0d1a6c3e870d06c928eaa124b3c58ed99afbc4b Mon Sep 17 00:00:00 2001 From: Jason Parraga Date: Fri, 22 Nov 2024 18:39:29 -0800 Subject: [PATCH 2/3] Address comments, fix issues, more tests Signed-off-by: Jason Parraga --- .../install/armadaserver_controller.go | 25 +- .../install/armadaserver_controller_test.go | 215 +++++++++++++++++- .../install/binoculars_controller.go | 27 +-- .../install/binoculars_controller_test.go | 214 ++++++++++++++++- internal/controller/install/common_helpers.go | 10 +- .../controller/install/executor_controller.go | 7 +- .../install/executor_controller_test.go | 169 ++++++++++++++ .../controller/install/lookout_controller.go | 9 +- .../install/scheduler_controller.go | 8 +- 9 files changed, 602 insertions(+), 82 deletions(-) diff --git a/internal/controller/install/armadaserver_controller.go b/internal/controller/install/armadaserver_controller.go index abd2e05..3dd642b 100644 --- a/internal/controller/install/armadaserver_controller.go +++ b/internal/controller/install/armadaserver_controller.go @@ -449,12 +449,7 @@ func createArmadaServerDeployment( volumeMounts := createVolumeMounts(GetConfigFilename(as.Name), as.Spec.AdditionalVolumeMounts) volumeMounts = append(volumeMounts, createPulsarVolumeMounts(pulsarConfig)...) - asConfig, err := extractArmadaServerConfig(as.Spec.ApplicationConfig) - if err != nil { - return nil, err - } - - readinessProbe, livenessProbe := CreateProbesWithScheme(GetServerScheme(asConfig.Grpc.Tls)) + readinessProbe, livenessProbe := CreateProbesWithScheme(GetServerScheme(commonConfig.GRPC.TLS)) deployment := appsv1.Deployment{ ObjectMeta: metav1.ObjectMeta{ @@ -687,21 +682,3 @@ func createServerPrometheusRule(server *installv1alpha1.ArmadaServer) *monitorin }, } } - -type ArmadaServerConfig struct { - Grpc GrpcConfig -} - -// extractArmadaServerConfig will unmarshal the appconfig and return the AramadaServerConfig portion -func extractArmadaServerConfig(config runtime.RawExtension) (ArmadaServerConfig, error) { - appConfig, err := builders.ConvertRawExtensionToYaml(config) - if err != nil { - return ArmadaServerConfig{}, err - } - var asConfig ArmadaServerConfig - err = yaml.Unmarshal([]byte(appConfig), &asConfig) - if err != nil { - return ArmadaServerConfig{}, err - } - return asConfig, err -} diff --git a/internal/controller/install/armadaserver_controller_test.go b/internal/controller/install/armadaserver_controller_test.go index e94f12e..e0c0956 100644 --- a/internal/controller/install/armadaserver_controller_test.go +++ b/internal/controller/install/armadaserver_controller_test.go @@ -5,6 +5,10 @@ import ( "testing" "time" + "github.com/google/go-cmp/cmp" + "google.golang.org/protobuf/testing/protocmp" + "k8s.io/apimachinery/pkg/util/intstr" + "github.com/armadaproject/armada-operator/internal/controller/builders" "k8s.io/utils/ptr" @@ -429,12 +433,12 @@ func TestSchedulerReconciler_createIngress(t *testing.T) { input := v1alpha1.ArmadaServer{ TypeMeta: metav1.TypeMeta{ - Kind: "Lookout", + Kind: "armadaserver", APIVersion: "install.armadaproject.io/v1alpha1", }, ObjectMeta: metav1.ObjectMeta{ Namespace: "default", - Name: "lookout", + Name: "armadaserver", }, Spec: v1alpha1.ArmadaServerSpec{ Replicas: ptr.To[int32](2), @@ -459,3 +463,210 @@ func TestSchedulerReconciler_createIngress(t *testing.T) { assert.NoError(t, err) assert.NotNil(t, ingress) } + +func TestArmadaServerReconciler_CreateDeployment(t *testing.T) { + t.Parallel() + + commonConfig := &builders.CommonApplicationConfig{ + HTTPPort: 8080, + GRPCPort: 5051, + MetricsPort: 9000, + Profiling: builders.ProfilingConfig{ + Port: 1337, + }, + } + + armadaServer := &installv1alpha1.ArmadaServer{ + TypeMeta: metav1.TypeMeta{ + Kind: "ArmadaServer", + APIVersion: "install.armadaproject.io/v1alpha1", + }, + ObjectMeta: metav1.ObjectMeta{Namespace: "default", Name: "armadaserver"}, + Spec: installv1alpha1.ArmadaServerSpec{ + PulsarInit: true, + CommonSpecBase: installv1alpha1.CommonSpecBase{ + Labels: map[string]string{"test": "hello"}, + Image: installv1alpha1.Image{ + Repository: "testrepo", + Tag: "1.0.0", + }, + ApplicationConfig: runtime.RawExtension{}, + Resources: &corev1.ResourceRequirements{}, + Prometheus: &installv1alpha1.PrometheusConfig{Enabled: true}, + }, + ClusterIssuer: "test", + HostNames: []string{"localhost"}, + Ingress: &installv1alpha1.IngressConfig{ + IngressClass: "nginx", + Labels: map[string]string{"test": "hello"}, + Annotations: map[string]string{"test": "hello"}, + }, + }, + } + + deployment, err := createArmadaServerDeployment(armadaServer, "armadaserver", commonConfig) + assert.NoError(t, err) + + expectedDeployment := &appsv1.Deployment{ + ObjectMeta: metav1.ObjectMeta{ + Name: "armadaserver", + Namespace: "default", + Labels: map[string]string{ + "app": "armadaserver", + "release": "armadaserver", + }, + }, + Spec: appsv1.DeploymentSpec{ + Replicas: ptr.To[int32](1), + Selector: &metav1.LabelSelector{ + MatchLabels: map[string]string{ + "app": "armadaserver", + }, + }, + Strategy: appsv1.DeploymentStrategy{ + Type: appsv1.RollingUpdateDeploymentStrategyType, + RollingUpdate: &appsv1.RollingUpdateDeployment{ + MaxUnavailable: &intstr.IntOrString{ + IntVal: 1, + }, + }, + }, + Template: corev1.PodTemplateSpec{ + ObjectMeta: metav1.ObjectMeta{ + Name: "armadaserver", + Namespace: "default", + Labels: map[string]string{ + "app": "armadaserver", + "release": "armadaserver", + }, + Annotations: map[string]string{ + "checksum/config": "e3b0c44298fc1c149afbf4c8996fb92427ae41e4649b934ca495991b7852b855", + }, + }, + Spec: corev1.PodSpec{ + Affinity: &corev1.Affinity{ + PodAffinity: &corev1.PodAffinity{ + PreferredDuringSchedulingIgnoredDuringExecution: []corev1.WeightedPodAffinityTerm{ + { + Weight: 100, + PodAffinityTerm: corev1.PodAffinityTerm{ + LabelSelector: &metav1.LabelSelector{ + MatchExpressions: []metav1.LabelSelectorRequirement{ + { + Key: "app", + Operator: metav1.LabelSelectorOpIn, + Values: []string{ + "armadaserver", + }, + }, + }, + }, + TopologyKey: "kubernetes.io/hostname", + }, + }, + }, + }, + }, + Volumes: []corev1.Volume{ + { + Name: "user-config", + VolumeSource: corev1.VolumeSource{ + Secret: &corev1.SecretVolumeSource{ + SecretName: "armadaserver", + }, + }, + }, + }, + Containers: []corev1.Container{ + { + Args: []string{ + "--config", + "/config/application_config.yaml", + }, + Env: []corev1.EnvVar{ + { + Name: "SERVICE_ACCOUNT", + ValueFrom: &corev1.EnvVarSource{ + FieldRef: &corev1.ObjectFieldSelector{ + FieldPath: "spec.serviceAccountName", + }, + }, + }, + { + Name: "POD_NAMESPACE", + ValueFrom: &corev1.EnvVarSource{ + FieldRef: &corev1.ObjectFieldSelector{ + FieldPath: "metadata.namespace", + }, + }, + }, + }, + Image: "testrepo:1.0.0", + ImagePullPolicy: corev1.PullIfNotPresent, + LivenessProbe: &corev1.Probe{ + ProbeHandler: corev1.ProbeHandler{ + HTTPGet: &corev1.HTTPGetAction{ + Path: "/health", + Port: intstr.FromString("http"), + Scheme: corev1.URISchemeHTTP, + }, + }, + InitialDelaySeconds: 10, + TimeoutSeconds: 10, + FailureThreshold: 3, + }, + Name: "armadaserver", + Ports: []corev1.ContainerPort{ + { + Name: "grpc", + ContainerPort: 5051, + Protocol: corev1.ProtocolTCP, + }, + { + Name: "http", + ContainerPort: 8080, + Protocol: corev1.ProtocolTCP, + }, + { + Name: "metrics", + ContainerPort: 9000, + Protocol: corev1.ProtocolTCP, + }, + { + Name: "profiling", + ContainerPort: 1337, + Protocol: corev1.ProtocolTCP, + }, + }, + ReadinessProbe: &corev1.Probe{ + ProbeHandler: corev1.ProbeHandler{ + HTTPGet: &corev1.HTTPGetAction{ + Path: "/health", + Port: intstr.FromString("http"), + Scheme: corev1.URISchemeHTTP, + }, + }, + InitialDelaySeconds: 5, + TimeoutSeconds: 5, + FailureThreshold: 2, + }, + VolumeMounts: []corev1.VolumeMount{ + { + Name: "user-config", + ReadOnly: true, + MountPath: appConfigFilepath, + SubPath: "armadaserver-config.yaml", + }, + }, + }, + }, + ServiceAccountName: "armadaserver", + }, + }, + }, + } + + if !cmp.Equal(expectedDeployment, deployment, protocmp.Transform()) { + t.Fatalf("deployment is not the same %s", cmp.Diff(expectedDeployment, deployment, protocmp.Transform())) + } +} diff --git a/internal/controller/install/binoculars_controller.go b/internal/controller/install/binoculars_controller.go index 5917347..95b328c 100644 --- a/internal/controller/install/binoculars_controller.go +++ b/internal/controller/install/binoculars_controller.go @@ -20,8 +20,6 @@ import ( "context" "time" - "sigs.k8s.io/yaml" - "github.com/pkg/errors" installv1alpha1 "github.com/armadaproject/armada-operator/api/install/v1alpha1" @@ -229,12 +227,7 @@ func createBinocularsDeployment( env := createEnv(binoculars.Spec.Environment) volumes := createVolumes(binoculars.Name, binoculars.Spec.AdditionalVolumes) volumeMounts := createVolumeMounts(GetConfigFilename(secret.Name), binoculars.Spec.AdditionalVolumeMounts) - - binocularsConfig, err := extractBinocularsConfig(binoculars.Spec.ApplicationConfig) - if err != nil { - return nil, err - } - readinessProbe, livenessProbe := CreateProbesWithScheme(GetServerScheme(binocularsConfig.Grpc.Tls)) + readinessProbe, livenessProbe := CreateProbesWithScheme(GetServerScheme(commonConfig.GRPC.TLS)) containers := []corev1.Container{{ Name: "binoculars", @@ -378,21 +371,3 @@ func (r *BinocularsReconciler) SetupWithManager(mgr ctrl.Manager) error { For(&installv1alpha1.Binoculars{}). Complete(r) } - -type BinocularsConfig struct { - Grpc GrpcConfig -} - -// extractBinocularsConfig will unmarshal the appconfig and return the BinocularsConfig portion -func extractBinocularsConfig(config runtime.RawExtension) (BinocularsConfig, error) { - appConfig, err := builders.ConvertRawExtensionToYaml(config) - if err != nil { - return BinocularsConfig{}, err - } - var binocularsConfig BinocularsConfig - err = yaml.Unmarshal([]byte(appConfig), &binocularsConfig) - if err != nil { - return BinocularsConfig{}, err - } - return binocularsConfig, err -} diff --git a/internal/controller/install/binoculars_controller_test.go b/internal/controller/install/binoculars_controller_test.go index 1049608..2e6f5d3 100644 --- a/internal/controller/install/binoculars_controller_test.go +++ b/internal/controller/install/binoculars_controller_test.go @@ -5,6 +5,10 @@ import ( "testing" "time" + "github.com/google/go-cmp/cmp" + "google.golang.org/protobuf/testing/protocmp" + "k8s.io/apimachinery/pkg/util/intstr" + "github.com/armadaproject/armada-operator/internal/controller/builders" "k8s.io/utils/ptr" @@ -14,6 +18,7 @@ import ( "github.com/armadaproject/armada-operator/test/k8sclient" "github.com/golang/mock/gomock" + appsv1 "k8s.io/api/apps/v1" v1 "k8s.io/api/apps/v1" corev1 "k8s.io/api/core/v1" networkingv1 "k8s.io/api/networking/v1" @@ -476,12 +481,12 @@ func TestSchedulerReconciler_createBinocularsIngress(t *testing.T) { input := v1alpha1.Binoculars{ TypeMeta: metav1.TypeMeta{ - Kind: "Lookout", + Kind: "binoculars", APIVersion: "install.armadaproject.io/v1alpha1", }, ObjectMeta: metav1.ObjectMeta{ Namespace: "default", - Name: "lookout", + Name: "binoculars", }, Spec: v1alpha1.BinocularsSpec{ Replicas: ptr.To[int32](2), @@ -506,3 +511,208 @@ func TestSchedulerReconciler_createBinocularsIngress(t *testing.T) { assert.NoError(t, err) assert.NotNil(t, ingress) } + +func TestBinocularsReconciler_CreateDeployment(t *testing.T) { + t.Parallel() + + commonConfig := &builders.CommonApplicationConfig{ + HTTPPort: 8080, + GRPCPort: 5051, + MetricsPort: 9000, + Profiling: builders.ProfilingConfig{ + Port: 1337, + }, + } + + secret := &corev1.Secret{ + ObjectMeta: metav1.ObjectMeta{ + Name: "binoculars", + }, + } + + binoculars := &v1alpha1.Binoculars{ + TypeMeta: metav1.TypeMeta{ + Kind: "Binoculars", + APIVersion: "install.armadaproject.io/v1alpha1", + }, + ObjectMeta: metav1.ObjectMeta{Namespace: "default", Name: "binoculars"}, + Spec: v1alpha1.BinocularsSpec{ + CommonSpecBase: installv1alpha1.CommonSpecBase{ + Labels: map[string]string{"test": "hello"}, + Image: installv1alpha1.Image{ + Repository: "testrepo", + Tag: "1.0.0", + }, + ApplicationConfig: runtime.RawExtension{}, + Resources: &corev1.ResourceRequirements{}, + }, + + Replicas: ptr.To[int32](2), + HostNames: []string{"localhost"}, + ClusterIssuer: "test", + Ingress: &installv1alpha1.IngressConfig{ + IngressClass: "nginx", + Labels: map[string]string{"test": "hello"}, + Annotations: map[string]string{"test": "hello"}, + }, + }, + } + + deployment, err := createBinocularsDeployment(binoculars, secret, "binoculars", commonConfig) + assert.NoError(t, err) + + expectedDeployment := &appsv1.Deployment{ + ObjectMeta: metav1.ObjectMeta{ + Name: "binoculars", + Namespace: "default", + Labels: map[string]string{ + "app": "binoculars", + "release": "binoculars", + }, + }, + Spec: appsv1.DeploymentSpec{ + Replicas: ptr.To[int32](2), + Selector: &metav1.LabelSelector{ + MatchLabels: map[string]string{ + "app": "binoculars", + }, + }, + Template: corev1.PodTemplateSpec{ + ObjectMeta: metav1.ObjectMeta{ + Name: "binoculars", + Namespace: "default", + Labels: map[string]string{ + "app": "binoculars", + "release": "binoculars", + }, + Annotations: map[string]string{ + "checksum/config": "e3b0c44298fc1c149afbf4c8996fb92427ae41e4649b934ca495991b7852b855", + }, + }, + Spec: corev1.PodSpec{ + Affinity: &corev1.Affinity{ + PodAffinity: &corev1.PodAffinity{ + PreferredDuringSchedulingIgnoredDuringExecution: []corev1.WeightedPodAffinityTerm{ + { + Weight: 100, + PodAffinityTerm: corev1.PodAffinityTerm{ + LabelSelector: &metav1.LabelSelector{ + MatchExpressions: []metav1.LabelSelectorRequirement{ + { + Key: "app", + Operator: metav1.LabelSelectorOpIn, + Values: []string{ + "binoculars", + }, + }, + }, + }, + TopologyKey: "kubernetes.io/hostname", + }, + }, + }, + }, + }, + Volumes: []corev1.Volume{ + { + Name: "user-config", + VolumeSource: corev1.VolumeSource{ + Secret: &corev1.SecretVolumeSource{ + SecretName: "binoculars", + }, + }, + }, + }, + Containers: []corev1.Container{ + { + Args: []string{ + "--config", + "/config/application_config.yaml", + }, + Env: []corev1.EnvVar{ + { + Name: "SERVICE_ACCOUNT", + ValueFrom: &corev1.EnvVarSource{ + FieldRef: &corev1.ObjectFieldSelector{ + FieldPath: "spec.serviceAccountName", + }, + }, + }, + { + Name: "POD_NAMESPACE", + ValueFrom: &corev1.EnvVarSource{ + FieldRef: &corev1.ObjectFieldSelector{ + FieldPath: "metadata.namespace", + }, + }, + }, + }, + Image: "testrepo:1.0.0", + ImagePullPolicy: corev1.PullIfNotPresent, + LivenessProbe: &corev1.Probe{ + ProbeHandler: corev1.ProbeHandler{ + HTTPGet: &corev1.HTTPGetAction{ + Path: "/health", + Port: intstr.FromString("http"), + Scheme: corev1.URISchemeHTTP, + }, + }, + InitialDelaySeconds: 10, + TimeoutSeconds: 10, + FailureThreshold: 3, + }, + Name: "binoculars", + Ports: []corev1.ContainerPort{ + { + Name: "grpc", + ContainerPort: 5051, + Protocol: corev1.ProtocolTCP, + }, + { + Name: "http", + ContainerPort: 8080, + Protocol: corev1.ProtocolTCP, + }, + { + Name: "metrics", + ContainerPort: 9000, + Protocol: corev1.ProtocolTCP, + }, + { + Name: "profiling", + ContainerPort: 1337, + Protocol: corev1.ProtocolTCP, + }, + }, + ReadinessProbe: &corev1.Probe{ + ProbeHandler: corev1.ProbeHandler{ + HTTPGet: &corev1.HTTPGetAction{ + Path: "/health", + Port: intstr.FromString("http"), + Scheme: corev1.URISchemeHTTP, + }, + }, + InitialDelaySeconds: 5, + TimeoutSeconds: 5, + FailureThreshold: 2, + }, + VolumeMounts: []corev1.VolumeMount{ + { + Name: "user-config", + ReadOnly: true, + MountPath: appConfigFilepath, + SubPath: "binoculars-config.yaml", + }, + }, + }, + }, + ServiceAccountName: "binoculars", + }, + }, + }, + } + + if !cmp.Equal(expectedDeployment, deployment, protocmp.Transform()) { + t.Fatalf("deployment is not the same %s", cmp.Diff(expectedDeployment, deployment, protocmp.Transform())) + } +} diff --git a/internal/controller/install/common_helpers.go b/internal/controller/install/common_helpers.go index 0b61f70..5d9e0ff 100644 --- a/internal/controller/install/common_helpers.go +++ b/internal/controller/install/common_helpers.go @@ -229,14 +229,6 @@ type PulsarConfig struct { Cacert string } -type GrpcConfig struct { - Tls TlsConfig -} - -type TlsConfig struct { - Enabled bool -} - // ArmadaInit used to initialize pulsar type ArmadaInit struct { Enabled bool @@ -329,7 +321,7 @@ func ExtractPulsarConfig(config runtime.RawExtension) (PulsarConfig, error) { } // GetServerScheme returns the URI scheme for the grpc server -func GetServerScheme(tlsConfig TlsConfig) corev1.URIScheme { +func GetServerScheme(tlsConfig builders.TLSConfig) corev1.URIScheme { if tlsConfig.Enabled { return corev1.URISchemeHTTPS } diff --git a/internal/controller/install/executor_controller.go b/internal/controller/install/executor_controller.go index 9fac5c0..3e79c0e 100644 --- a/internal/controller/install/executor_controller.go +++ b/internal/controller/install/executor_controller.go @@ -180,7 +180,7 @@ func (r *ExecutorReconciler) generateExecutorInstallComponents( } serviceAccountName = serviceAccount.Name } - deployment := r.createDeployment(executor, serviceAccountName, config) + deployment := createExecutorDeployment(executor, serviceAccountName, config) if err = controllerutil.SetOwnerReference(executor, deployment, scheme); err != nil { return nil, errors.WithStack(err) } @@ -241,7 +241,7 @@ func (r *ExecutorReconciler) generateExecutorInstallComponents( return components, nil } -func (r *ExecutorReconciler) createDeployment( +func createExecutorDeployment( executor *installv1alpha1.Executor, serviceAccountName string, config *builders.CommonApplicationConfig, @@ -249,7 +249,6 @@ func (r *ExecutorReconciler) createDeployment( var replicas int32 = 1 volumes := createVolumes(executor.Name, executor.Spec.AdditionalVolumes) volumeMounts := createVolumeMounts(GetConfigFilename(executor.Name), executor.Spec.AdditionalVolumeMounts) - readinessProbe, livenessProbe := CreateProbesWithScheme(corev1.URISchemeHTTP) env := []corev1.EnvVar{ @@ -276,7 +275,7 @@ func (r *ExecutorReconciler) createDeployment( ImagePullPolicy: corev1.PullIfNotPresent, Image: ImageString(executor.Spec.Image), Args: []string{appConfigFlag, appConfigFilepath}, - Ports: newContainerPortsMetrics(config), + Ports: newContainerPortsHTTPWithMetrics(config), Env: env, VolumeMounts: volumeMounts, SecurityContext: executor.Spec.SecurityContext, diff --git a/internal/controller/install/executor_controller_test.go b/internal/controller/install/executor_controller_test.go index 0d9a480..830d4ca 100644 --- a/internal/controller/install/executor_controller_test.go +++ b/internal/controller/install/executor_controller_test.go @@ -5,6 +5,13 @@ import ( "testing" "time" + "github.com/google/go-cmp/cmp" + "google.golang.org/protobuf/testing/protocmp" + "k8s.io/apimachinery/pkg/util/intstr" + "k8s.io/utils/ptr" + + "github.com/armadaproject/armada-operator/internal/controller/builders" + monitoringv1 "github.com/prometheus-operator/prometheus-operator/pkg/apis/monitoring/v1" "github.com/armadaproject/armada-operator/test/k8sclient" @@ -394,3 +401,165 @@ func TestExecutorReconciler_generateAdditionalClusterRoles(t *testing.T) { } assert.Equal(t, expectedClusterRoleBinding2, *bindings[1]) } + +func TestExecutorReconciler_CreateDeployment(t *testing.T) { + t.Parallel() + + commonConfig := &builders.CommonApplicationConfig{ + HTTPPort: 8080, + GRPCPort: 5051, + MetricsPort: 9000, + Profiling: builders.ProfilingConfig{ + Port: 1337, + }, + } + + executor := &installv1alpha1.Executor{ + TypeMeta: metav1.TypeMeta{ + Kind: "Executor", + APIVersion: "install.armadaproject.io/v1alpha1", + }, + ObjectMeta: metav1.ObjectMeta{Namespace: "default", Name: "executor"}, + Spec: installv1alpha1.ExecutorSpec{ + CommonSpecBase: installv1alpha1.CommonSpecBase{ + Labels: nil, + Image: installv1alpha1.Image{ + Repository: "testrepo", + Tag: "1.0.0", + }, + ApplicationConfig: runtime.RawExtension{}, + Resources: &corev1.ResourceRequirements{}, + Prometheus: &installv1alpha1.PrometheusConfig{Enabled: true, ScrapeInterval: &metav1.Duration{Duration: 1 * time.Second}}, + }, + }, + } + + deployment := createExecutorDeployment(executor, "executor", commonConfig) + + expectedDeployment := &appsv1.Deployment{ + ObjectMeta: metav1.ObjectMeta{ + Name: "executor", + Namespace: "default", + Labels: map[string]string{ + "app": "executor", + "release": "executor", + }, + }, + Spec: appsv1.DeploymentSpec{ + Replicas: ptr.To[int32](1), + Selector: &metav1.LabelSelector{ + MatchLabels: map[string]string{ + "app": "executor", + }, + }, + Template: corev1.PodTemplateSpec{ + ObjectMeta: metav1.ObjectMeta{ + Name: "executor", + Namespace: "default", + Labels: map[string]string{ + "app": "executor", + "release": "executor", + }, + Annotations: map[string]string{ + "checksum/config": "e3b0c44298fc1c149afbf4c8996fb92427ae41e4649b934ca495991b7852b855", + }, + }, + Spec: corev1.PodSpec{ + Volumes: []corev1.Volume{ + { + Name: "user-config", + VolumeSource: corev1.VolumeSource{ + Secret: &corev1.SecretVolumeSource{ + SecretName: "executor", + }, + }, + }, + }, + Containers: []corev1.Container{ + { + Args: []string{ + "--config", + "/config/application_config.yaml", + }, + Env: []corev1.EnvVar{ + { + Name: "SERVICE_ACCOUNT", + ValueFrom: &corev1.EnvVarSource{ + FieldRef: &corev1.ObjectFieldSelector{ + FieldPath: "spec.serviceAccountName", + }, + }, + }, + { + Name: "POD_NAMESPACE", + ValueFrom: &corev1.EnvVarSource{ + FieldRef: &corev1.ObjectFieldSelector{ + FieldPath: "metadata.namespace", + }, + }, + }, + }, + Image: "testrepo:1.0.0", + ImagePullPolicy: corev1.PullIfNotPresent, + LivenessProbe: &corev1.Probe{ + ProbeHandler: corev1.ProbeHandler{ + HTTPGet: &corev1.HTTPGetAction{ + Path: "/health", + Port: intstr.FromString("http"), + Scheme: corev1.URISchemeHTTP, + }, + }, + InitialDelaySeconds: 10, + TimeoutSeconds: 10, + FailureThreshold: 3, + }, + Name: "executor", + Ports: []corev1.ContainerPort{ + { + Name: "http", + ContainerPort: 8080, + Protocol: corev1.ProtocolTCP, + }, + { + Name: "metrics", + ContainerPort: 9000, + Protocol: corev1.ProtocolTCP, + }, + { + Name: "profiling", + ContainerPort: 1337, + Protocol: corev1.ProtocolTCP, + }, + }, + ReadinessProbe: &corev1.Probe{ + ProbeHandler: corev1.ProbeHandler{ + HTTPGet: &corev1.HTTPGetAction{ + Path: "/health", + Port: intstr.FromString("http"), + Scheme: corev1.URISchemeHTTP, + }, + }, + InitialDelaySeconds: 5, + TimeoutSeconds: 5, + FailureThreshold: 2, + }, + VolumeMounts: []corev1.VolumeMount{ + { + Name: "user-config", + ReadOnly: true, + MountPath: appConfigFilepath, + SubPath: "executor-config.yaml", + }, + }, + }, + }, + ServiceAccountName: "executor", + }, + }, + }, + } + + if !cmp.Equal(expectedDeployment, deployment, protocmp.Transform()) { + t.Fatalf("deployment is not the same %s", cmp.Diff(expectedDeployment, deployment, protocmp.Transform())) + } +} diff --git a/internal/controller/install/lookout_controller.go b/internal/controller/install/lookout_controller.go index 82f7f3b..79a6c0f 100644 --- a/internal/controller/install/lookout_controller.go +++ b/internal/controller/install/lookout_controller.go @@ -139,7 +139,6 @@ func (r *LookoutReconciler) Reconcile(ctx context.Context, req ctrl.Request) (ct type LookoutConfig struct { Postgres PostgresConfig - Tls TlsConfig } func generateLookoutInstallComponents( @@ -271,13 +270,7 @@ func createLookoutDeployment(lookout *installv1alpha1.Lookout, serviceAccountNam env := createEnv(lookout.Spec.Environment) volumes := createVolumes(lookout.Name, lookout.Spec.AdditionalVolumes) volumeMounts := createVolumeMounts(GetConfigFilename(lookout.Name), lookout.Spec.AdditionalVolumeMounts) - - lookoutConfig, err := extractLookoutConfig(lookout.Spec.ApplicationConfig) - if err != nil { - return nil, err - } - - readinessProbe, livenessProbe := CreateProbesWithScheme(GetServerScheme(lookoutConfig.Tls)) + readinessProbe, livenessProbe := CreateProbesWithScheme(GetServerScheme(config.GRPC.TLS)) deployment := appsv1.Deployment{ ObjectMeta: metav1.ObjectMeta{Name: lookout.Name, Namespace: lookout.Namespace, Labels: AllLabels(lookout.Name, lookout.Labels)}, diff --git a/internal/controller/install/scheduler_controller.go b/internal/controller/install/scheduler_controller.go index cae19a9..a221926 100644 --- a/internal/controller/install/scheduler_controller.go +++ b/internal/controller/install/scheduler_controller.go @@ -141,7 +141,6 @@ func (r *SchedulerReconciler) Reconcile(ctx context.Context, req ctrl.Request) ( } type SchedulerConfig struct { - Grpc GrpcConfig Postgres PostgresConfig } @@ -290,12 +289,7 @@ func newSchedulerDeployment( volumes = append(volumes, createPulsarVolumes(pulsarConfig)...) volumeMounts := createVolumeMounts(GetConfigFilename(scheduler.Name), scheduler.Spec.AdditionalVolumeMounts) volumeMounts = append(volumeMounts, createPulsarVolumeMounts(pulsarConfig)...) - - schedulerConfig, err := extractSchedulerConfig(scheduler.Spec.ApplicationConfig) - if err != nil { - return nil, err - } - readinessProbe, livenessProbe := CreateProbesWithScheme(GetServerScheme(schedulerConfig.Grpc.Tls)) + readinessProbe, livenessProbe := CreateProbesWithScheme(GetServerScheme(config.GRPC.TLS)) deployment := appsv1.Deployment{ ObjectMeta: metav1.ObjectMeta{Name: scheduler.Name, Namespace: scheduler.Namespace, Labels: AllLabels(scheduler.Name, scheduler.Labels)}, From 90e9d4572bf33a8c297240d238f9c1e5ec29ef53 Mon Sep 17 00:00:00 2001 From: Jason Parraga Date: Mon, 25 Nov 2024 20:56:25 -0800 Subject: [PATCH 3/3] Address comments Signed-off-by: Jason Parraga --- internal/controller/install/armadaserver_controller_test.go | 2 +- internal/controller/install/binoculars_controller_test.go | 2 +- internal/controller/install/common_helpers.go | 2 +- 3 files changed, 3 insertions(+), 3 deletions(-) diff --git a/internal/controller/install/armadaserver_controller_test.go b/internal/controller/install/armadaserver_controller_test.go index e0c0956..bd0319c 100644 --- a/internal/controller/install/armadaserver_controller_test.go +++ b/internal/controller/install/armadaserver_controller_test.go @@ -433,7 +433,7 @@ func TestSchedulerReconciler_createIngress(t *testing.T) { input := v1alpha1.ArmadaServer{ TypeMeta: metav1.TypeMeta{ - Kind: "armadaserver", + Kind: "ArmadaServer", APIVersion: "install.armadaproject.io/v1alpha1", }, ObjectMeta: metav1.ObjectMeta{ diff --git a/internal/controller/install/binoculars_controller_test.go b/internal/controller/install/binoculars_controller_test.go index 2e6f5d3..651c984 100644 --- a/internal/controller/install/binoculars_controller_test.go +++ b/internal/controller/install/binoculars_controller_test.go @@ -481,7 +481,7 @@ func TestSchedulerReconciler_createBinocularsIngress(t *testing.T) { input := v1alpha1.Binoculars{ TypeMeta: metav1.TypeMeta{ - Kind: "binoculars", + Kind: "Binoculars", APIVersion: "install.armadaproject.io/v1alpha1", }, ObjectMeta: metav1.ObjectMeta{ diff --git a/internal/controller/install/common_helpers.go b/internal/controller/install/common_helpers.go index 5d9e0ff..efc353f 100644 --- a/internal/controller/install/common_helpers.go +++ b/internal/controller/install/common_helpers.go @@ -317,7 +317,7 @@ func ExtractPulsarConfig(config runtime.RawExtension) (PulsarConfig, error) { if err != nil { return PulsarConfig{}, err } - return asConfig.Pulsar, err + return asConfig.Pulsar, nil } // GetServerScheme returns the URI scheme for the grpc server