blob: a26db2fbe5e2d1373651448a21123aab91336ec8 [file] [log] [blame]
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you 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 controllers
import (
"context"
"fmt"
"time"
"github.com/apache/incubator-kie-kogito-serverless-operator/api"
operatorapi "github.com/apache/incubator-kie-kogito-serverless-operator/api/v1alpha08"
clientr "github.com/apache/incubator-kie-kogito-serverless-operator/container-builder/client"
"github.com/apache/incubator-kie-kogito-serverless-operator/controllers/clusterplatform"
"github.com/apache/incubator-kie-kogito-serverless-operator/controllers/platform"
"github.com/apache/incubator-kie-kogito-serverless-operator/controllers/platform/services"
"github.com/apache/incubator-kie-kogito-serverless-operator/log"
appsv1 "k8s.io/api/apps/v1"
corev1 "k8s.io/api/core/v1"
"k8s.io/apimachinery/pkg/api/errors"
"k8s.io/apimachinery/pkg/runtime"
"k8s.io/apimachinery/pkg/types"
"k8s.io/client-go/rest"
"k8s.io/client-go/tools/record"
"k8s.io/klog/v2"
ctrlrun "sigs.k8s.io/controller-runtime"
"sigs.k8s.io/controller-runtime/pkg/client"
ctrl "sigs.k8s.io/controller-runtime/pkg/client"
"sigs.k8s.io/controller-runtime/pkg/handler"
"sigs.k8s.io/controller-runtime/pkg/reconcile"
)
// SonataFlowPlatformReconciler reconciles a SonataFlowPlatform object
type SonataFlowPlatformReconciler struct {
// This Client, initialized using mgr.Client() above, is a split Client
// that reads objects from the cache and writes to the API server
ctrl.Client
// Non-caching Client
Reader ctrl.Reader
Scheme *runtime.Scheme
Config *rest.Config
Recorder record.EventRecorder
}
//+kubebuilder:rbac:groups=sonataflow.org,resources=sonataflowplatforms,verbs=get;list;watch;create;update;patch;delete
//+kubebuilder:rbac:groups=sonataflow.org,resources=sonataflowplatforms/status,verbs=get;update;patch
//+kubebuilder:rbac:groups=sonataflow.org,resources=sonataflowplatforms/finalizers,verbs=update
// Reconcile is part of the main kubernetes reconciliation loop which aims to
// move the current state of the cluster closer to the desired state.
// the SonataFlowPlatform object against the actual cluster state, and then
// perform operations to make the cluster state reflect the state specified by
// the user.
//
// For more details, check Reconcile and its Result here:
// - https://pkg.go.dev/sigs.k8s.io/controller-runtime@v0.12.1/pkg/reconcile
func (r *SonataFlowPlatformReconciler) Reconcile(ctx context.Context, req reconcile.Request) (reconcile.Result, error) {
// Make sure the operator is allowed to act on namespace
if ok, err := platform.IsOperatorAllowedOnNamespace(ctx, r.Reader, req.Namespace); err != nil {
return reconcile.Result{}, err
} else if !ok {
klog.V(log.I).InfoS("Ignoring request because the operator hasn't got the permissions to work on namespace", "namespace", req.Namespace)
return reconcile.Result{}, nil
}
// Fetch the Platform instance
var instance operatorapi.SonataFlowPlatform
if err := r.Reader.Get(ctx, req.NamespacedName, &instance); err != nil {
if errors.IsNotFound(err) {
// Request object not found, could have been deleted after reconcile request.
// Owned objects are automatically garbage collected. For additional cleanup
// logic use finalizers.
// Return and don't requeue
return reconcile.Result{}, nil
}
// Error reading the object - requeue the request.
return reconcile.Result{}, err
}
instance.Status.Manager().InitializeConditions()
// Only process resources assigned to the operator
if !platform.IsOperatorHandlerConsideringLock(ctx, r.Reader, req.Namespace, &instance) {
klog.V(log.I).InfoS("Ignoring request because resource is not assigned to current operator")
return reconcile.Result{}, nil
}
actions := []platform.Action{
platform.NewInitializeAction(),
platform.NewServiceAction(),
platform.NewWarmAction(r.Reader),
platform.NewCreateAction(),
platform.NewMonitorAction(),
}
var err error
target := instance.DeepCopy()
if err = r.SonataFlowPlatformUpdateStatus(ctx, req, target); err != nil {
return reconcile.Result{}, err
}
for _, a := range actions {
cli, _ := clientr.FromCtrlClientSchemeAndConfig(r.Client, r.Scheme, r.Config)
a.InjectClient(cli)
if a.CanHandle(target) {
klog.V(log.I).InfoS("Invoking action", "Name", a.Name())
target, err = a.Handle(ctx, target)
if err != nil {
if target != nil {
target.Status.Manager().MarkFalse(api.SucceedConditionType, operatorapi.PlatformFailureReason, err.Error())
if err := r.Client.Status().Patch(ctx, target, ctrl.MergeFrom(&instance)); err != nil {
return reconcile.Result{}, err
}
}
r.Recorder.Event(&instance, corev1.EventTypeWarning, "Failed", fmt.Sprintf("Failed to update SonataFlowPlaform: %s", err))
return reconcile.Result{}, err
}
if target != nil {
target.Status.ObservedGeneration = instance.Generation
if err := r.Client.Status().Patch(ctx, target, ctrl.MergeFrom(&instance)); err != nil {
r.Recorder.Event(&instance, corev1.EventTypeNormal, "Status Updated", fmt.Sprintf("Updated platform condition %s", instance.Status.GetTopLevelCondition()))
return reconcile.Result{}, err
}
if err := r.Client.Update(ctx, target); err != nil {
r.Recorder.Event(&instance, corev1.EventTypeNormal, "Spec Updated", fmt.Sprintf("Updated platform condition to %s", instance.Status.GetTopLevelCondition()))
return reconcile.Result{}, err
}
}
// handle one action at time so the resource
// is always at its latest state
r.Recorder.Event(&instance, corev1.EventTypeNormal, "Updated", fmt.Sprintf("Updated platform condition to %s", instance.Status.GetTopLevelCondition()))
break
}
}
if target != nil && target.Status.IsReady() {
return reconcile.Result{}, nil
}
// Requeue
return reconcile.Result{
RequeueAfter: 5 * time.Second,
}, nil
}
// If an active cluster platform exists, update platform.Status accordingly
func (r *SonataFlowPlatformReconciler) SonataFlowPlatformUpdateStatus(ctx context.Context, req reconcile.Request, target *operatorapi.SonataFlowPlatform) error {
// Fetch the active SonataFlowClusterPlatform instance
sfcPlatform, err := clusterplatform.GetActiveClusterPlatform(ctx, r.Client)
if err != nil && !errors.IsNotFound(err) {
klog.V(log.E).ErrorS(err, "Failed to get active SonataFlowClusterPlatform")
return err
}
if sfcPlatform != nil {
sfPlatform := &operatorapi.SonataFlowPlatform{}
platformRef := sfcPlatform.Spec.PlatformRef
namespacedName := types.NamespacedName{Namespace: platformRef.Namespace, Name: platformRef.Name}
if req.NamespacedName == namespacedName {
sfPlatform = target.DeepCopy()
} else {
// retrieve referenced platform object
err := r.Reader.Get(ctx, namespacedName, sfPlatform)
if err != nil && !errors.IsNotFound(err) {
klog.V(log.E).ErrorS(err, "Failed to get referenced SonataFlowPlatform", namespacedName)
return err
}
}
target.Status.ClusterPlatformRef = &operatorapi.SonataFlowClusterPlatformRefStatus{
Name: sfcPlatform.Name,
PlatformRef: operatorapi.SonataFlowPlatformRef{
Name: platformRef.Name,
Namespace: platformRef.Namespace,
},
}
if sfcPlatform.Spec.Capabilities != nil && contains(sfcPlatform.Spec.Capabilities.Workflows, clusterplatform.PlatformServices) {
tpsDI := services.NewDataIndexHandler(target)
tpsDI.SetServiceUrlInStatus(sfPlatform)
tpsJS := services.NewJobServiceHandler(target)
tpsJS.SetServiceUrlInStatus(sfPlatform)
}
} else {
target.Status.ClusterPlatformRef = nil
}
return nil
}
// SetupWithManager sets up the controller with the Manager.
func (r *SonataFlowPlatformReconciler) SetupWithManager(mgr ctrlrun.Manager) error {
return ctrlrun.NewControllerManagedBy(mgr).
For(&operatorapi.SonataFlowPlatform{}).
Owns(&appsv1.Deployment{}).
Owns(&corev1.Service{}).
Owns(&corev1.ConfigMap{}).
Watches(&operatorapi.SonataFlowPlatform{}, handler.EnqueueRequestsFromMapFunc(r.mapPlatformToPlatformRequests)).
Watches(&operatorapi.SonataFlowClusterPlatform{}, handler.EnqueueRequestsFromMapFunc(r.mapClusterPlatformToPlatformRequests)).
Complete(r)
}
// if active clusterplatform object is changed, reconcile all SonataFlowPlatforms in the cluster.
func (r *SonataFlowPlatformReconciler) mapClusterPlatformToPlatformRequests(ctx context.Context, object client.Object) []reconcile.Request {
sfcPlatform := object.(*operatorapi.SonataFlowClusterPlatform)
if sfcPlatform != nil && clusterplatform.IsActive(sfcPlatform) {
return r.platformRequests(ctx, sfcPlatform, true)
}
return nil
}
// if actively referenced sonataflowplatform is changed, reconcile other SonataFlowPlatforms in the cluster.
func (r *SonataFlowPlatformReconciler) mapPlatformToPlatformRequests(ctx context.Context, object client.Object) []reconcile.Request {
platform := object.(*operatorapi.SonataFlowPlatform)
sfcPlatform, err := clusterplatform.GetActiveClusterPlatform(ctx, r.Client)
if err != nil && !errors.IsNotFound(err) {
klog.V(log.E).ErrorS(err, "Failed to get active SonataFlowClusterPlatform")
return nil
}
if sfcPlatform != nil {
sfpcRefNsName := types.NamespacedName{Namespace: sfcPlatform.Spec.PlatformRef.Namespace, Name: sfcPlatform.Spec.PlatformRef.Name}
if client.ObjectKeyFromObject(platform) == sfpcRefNsName {
return r.platformRequests(ctx, sfcPlatform, false)
}
}
return nil
}
func (r *SonataFlowPlatformReconciler) platformRequests(ctx context.Context, sfcPlatform *operatorapi.SonataFlowClusterPlatform, allPlatforms bool) []reconcile.Request {
var plList operatorapi.SonataFlowPlatformList
if err := r.List(ctx, &plList, client.InNamespace("")); err != nil {
klog.V(log.E).ErrorS(err, "could not list SonataFlowPlatforms. "+
"SonataFlowPlatforms affected by changes to the active SonataFlowPlatform or SonataFlowClusterPlatform object will not be reconciled.")
return nil
}
sfpcRefNsName := types.NamespacedName{Namespace: sfcPlatform.Spec.PlatformRef.Namespace, Name: sfcPlatform.Spec.PlatformRef.Name}
var requests []reconcile.Request
for _, platform := range plList.Items {
sfpNsName := client.ObjectKeyFromObject(&platform)
// this check is required so that the cluster-referenced platform object doesn't infinitely reconcile
if sfpNsName != sfpcRefNsName || allPlatforms {
requests = append(requests, reconcile.Request{NamespacedName: sfpNsName})
}
}
return requests
}
func contains(slice []operatorapi.WorkFlowCapability, s operatorapi.WorkFlowCapability) bool {
for _, a := range slice {
if a == s {
return true
}
}
return false
}