blob: f9c3556557d28f2c7ca6391f766b9df2bac63424 [file] [log] [blame]
// Copyright 2023 Red Hat, Inc. and/or its affiliates
//
// 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 profiles
import (
"github.com/magiconair/properties"
appsv1 "k8s.io/api/apps/v1"
corev1 "k8s.io/api/core/v1"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
"k8s.io/apimachinery/pkg/util/intstr"
"sigs.k8s.io/controller-runtime/pkg/client"
"sigs.k8s.io/controller-runtime/pkg/controller/controllerutil"
"github.com/kiegroup/kogito-serverless-operator/workflowproj"
operatorapi "github.com/kiegroup/kogito-serverless-operator/api/v1alpha08"
kubeutil "github.com/kiegroup/kogito-serverless-operator/utils/kubernetes"
"github.com/kiegroup/kogito-serverless-operator/utils/openshift"
)
const (
defaultHTTPWorkflowPortInt = 8080
defaultHTTPWorkflowPortStr = "8080"
defaultContainerName = "workflow"
defaultHTTPServicePort = 80
configMapWorkflowPropsVolumeName = "workflow-properties"
// Quarkus Health Check Probe configuration.
// See: https://quarkus.io/guides/smallrye-health#running-the-health-check
quarkusHealthPathStarted = "/q/health/started"
quarkusHealthPathReady = "/q/health/ready"
quarkusHealthPathLive = "/q/health/live"
// Default deployment health check configuration
// See: https://kubernetes.io/docs/tasks/configure-pod-container/configure-liveness-readiness-startup-probes/
healthTimeoutSeconds = 3
healthStartedFailureThreshold = 5
healthStartedPeriodSeconds = 15
healthStartedInitialDelaySeconds = 10
)
var defaultHTTPWorkflowPortIntStr = intstr.FromInt(defaultHTTPWorkflowPortInt)
// same for now
var defaultProdApplicationProperties = defaultDevApplicationProperties
// objectCreator is the func that creates the initial reference object, if the object doesn't exist in the cluster, this one is created.
// Can be used as a reference to keep the object immutable
type objectCreator func(workflow *operatorapi.SonataFlow) (client.Object, error)
// defaultDeploymentCreator is an objectCreator for a base Kubernetes Deployments for profiles that need to deploy the workflow on a vanilla deployment.
// It serves as a basis for a basic Quarkus Java application, expected to listen on http 8080.
func defaultDeploymentCreator(workflow *operatorapi.SonataFlow) (client.Object, error) {
lbl := workflowproj.GetDefaultLabels(workflow)
size := int32(1)
deployment := &appsv1.Deployment{
ObjectMeta: metav1.ObjectMeta{
Name: workflow.Name,
Namespace: workflow.Namespace,
Labels: lbl,
},
Spec: appsv1.DeploymentSpec{
Replicas: &size,
Selector: &metav1.LabelSelector{
MatchLabels: lbl,
},
Template: corev1.PodTemplateSpec{
ObjectMeta: metav1.ObjectMeta{
Labels: lbl,
},
Spec: corev1.PodSpec{
Containers: []corev1.Container{{
Name: defaultContainerName,
Ports: []corev1.ContainerPort{{
ContainerPort: defaultHTTPWorkflowPortInt,
Name: "http",
Protocol: corev1.ProtocolTCP,
}},
TerminationMessagePolicy: corev1.TerminationMessageFallbackToLogsOnError,
LivenessProbe: &corev1.Probe{
ProbeHandler: corev1.ProbeHandler{
HTTPGet: &corev1.HTTPGetAction{
Path: quarkusHealthPathLive,
Port: defaultHTTPWorkflowPortIntStr,
},
},
TimeoutSeconds: healthTimeoutSeconds,
},
ReadinessProbe: &corev1.Probe{
ProbeHandler: corev1.ProbeHandler{
HTTPGet: &corev1.HTTPGetAction{
Path: quarkusHealthPathReady,
Port: defaultHTTPWorkflowPortIntStr,
},
},
TimeoutSeconds: healthTimeoutSeconds,
},
StartupProbe: &corev1.Probe{
ProbeHandler: corev1.ProbeHandler{
HTTPGet: &corev1.HTTPGetAction{
Path: quarkusHealthPathStarted,
Port: defaultHTTPWorkflowPortIntStr,
},
},
InitialDelaySeconds: healthStartedInitialDelaySeconds,
TimeoutSeconds: healthTimeoutSeconds,
FailureThreshold: healthStartedFailureThreshold,
PeriodSeconds: healthStartedPeriodSeconds,
},
ImagePullPolicy: corev1.PullAlways,
SecurityContext: kubeutil.SecurityDefaults(),
}},
},
},
},
}
return deployment, nil
}
// naiveApplyImageDeploymentMutateVisitor creates a visitor that mutates a vanilla Kubernetes Deployment to apply the given image in the first container
func naiveApplyImageDeploymentMutateVisitor(image string) mutateVisitor {
return func(object client.Object) controllerutil.MutateFn {
return func() error {
object.(*appsv1.Deployment).Spec.Template.Spec.Containers[0].Image = image
return nil
}
}
}
// defaultDeploymentMutateVisitor guarantees the state of the default Deployment object
func defaultDeploymentMutateVisitor(workflow *operatorapi.SonataFlow) mutateVisitor {
return func(object client.Object) controllerutil.MutateFn {
return func() error {
if kubeutil.IsObjectNew(object) {
return nil
}
original, err := defaultDeploymentCreator(workflow)
if err != nil {
return err
}
ensureDefaultDeployment(original.(*appsv1.Deployment), object.(*appsv1.Deployment))
return nil
}
}
}
// ensureDefaultDeployment ensure that the original Deployment fields are immutable.
func ensureDefaultDeployment(original *appsv1.Deployment, object *appsv1.Deployment) {
object.Spec.Replicas = original.Spec.Replicas
object.Spec.Selector = original.Spec.Selector
object.Labels = original.GetLabels()
workflowContainer := kubeutil.GetContainerByName(defaultContainerName, object)
if workflowContainer == nil {
object.Spec.Template.Spec.Containers = make([]corev1.Container, 0)
object.Spec.Template.Spec.Containers = original.Spec.Template.Spec.Containers
} else {
originalContainer := original.Spec.Template.Spec.Containers[0]
workflowContainer.SecurityContext = originalContainer.SecurityContext
workflowContainer.Ports = originalContainer.Ports
object.Spec.Template.Spec.Containers = make([]corev1.Container, 0)
object.Spec.Template.Spec.Containers = append(object.Spec.Template.Spec.Containers, *workflowContainer)
}
}
// defaultServiceCreator is an objectCreator for a basic Service aiming a vanilla Kubernetes Deployment.
// It maps the default HTTP port (80) to the target Java application webserver on port 8080.
func defaultServiceCreator(workflow *operatorapi.SonataFlow) (client.Object, error) {
lbl := workflowproj.GetDefaultLabels(workflow)
service := &corev1.Service{
ObjectMeta: metav1.ObjectMeta{
Name: workflow.Name,
Namespace: workflow.Namespace,
Labels: lbl,
},
Spec: corev1.ServiceSpec{
Selector: lbl,
Ports: []corev1.ServicePort{{
Protocol: corev1.ProtocolTCP,
Port: defaultHTTPServicePort,
TargetPort: defaultHTTPWorkflowPortIntStr,
}},
},
}
return service, nil
}
// defaultNetworkCreator is an objectCreator for a basic Route for a workflow using dev profile
// running on OpenShift.
// It enables the exposition of the dev service using an OpenShift Route.
// See: https://github.com/openshift/api/blob/d170fcdc0fa638b664e4f35f2daf753cb4afe36b/route/v1/route.crd.yaml
func defaultNetworkCreator(workflow *operatorapi.SonataFlow) (client.Object, error) {
route, err := openshift.RouteForWorkflow(workflow)
return route, err
}
func defaultServiceMutateVisitor(workflow *operatorapi.SonataFlow) mutateVisitor {
return func(object client.Object) controllerutil.MutateFn {
return func() error {
if kubeutil.IsObjectNew(object) {
return nil
}
original, err := defaultServiceCreator(workflow)
if err != nil {
return err
}
object.(*corev1.Service).Spec.Ports = original.(*corev1.Service).Spec.Ports
object.(*corev1.Service).Labels = original.GetLabels()
return nil
}
}
}
func ensureWorkflowPropertiesConfigMapMutator(workflow *operatorapi.SonataFlow, defaultProperties string) mutateVisitor {
return func(object client.Object) controllerutil.MutateFn {
return func() error {
if kubeutil.IsObjectNew(object) {
return nil
}
original, err := workflowPropsConfigMapCreator(workflow)
if err != nil {
return err
}
cm := object.(*corev1.ConfigMap)
cm.Labels = original.GetLabels()
_, hasKey := cm.Data[workflowproj.ApplicationPropertiesFileName]
if !hasKey {
cm.Data = make(map[string]string, 1)
cm.Data[workflowproj.ApplicationPropertiesFileName] = defaultProperties
} else {
props, propErr := properties.LoadString(cm.Data[workflowproj.ApplicationPropertiesFileName])
if propErr != nil {
// can't load user's properties, replace with default
cm.Data[workflowproj.ApplicationPropertiesFileName] = defaultProperties
return nil
}
originalProps := properties.MustLoadString(original.(*corev1.ConfigMap).Data[workflowproj.ApplicationPropertiesFileName])
// we overwrite with the defaults
props.Merge(originalProps)
// Disable expansions since it's not our responsibility
// Property expansion means resolving ${} within the properties and environment context. Quarkus will do that in runtime.
props.DisableExpansion = true
cm.Data[workflowproj.ApplicationPropertiesFileName] = props.String()
}
return nil
}
}
}
func ensureProdWorkflowPropertiesConfigMapMutator(workflow *operatorapi.SonataFlow) mutateVisitor {
return ensureWorkflowPropertiesConfigMapMutator(workflow, defaultProdApplicationProperties)
}
// workflowPropsConfigMapCreator creates a ConfigMap to hold the external application properties
func workflowPropsConfigMapCreator(workflow *operatorapi.SonataFlow) (client.Object, error) {
return workflowproj.CreateNewAppPropsConfigMap(workflow, defaultDevApplicationProperties), nil
}