/* * Please refer to the LICENSE file in the root directory of the project. * https://github.com/kubesphere/kubesphere/blob/master/LICENSE */ package workspace import ( "context" "github.com/go-logr/logr" corev1 "k8s.io/api/core/v1" "k8s.io/client-go/tools/record" "k8s.io/klog/v2" tenantv1beta1 "kubesphere.io/api/tenant/v1beta1" ctrl "sigs.k8s.io/controller-runtime" "sigs.k8s.io/controller-runtime/pkg/client" "sigs.k8s.io/controller-runtime/pkg/controller" "sigs.k8s.io/controller-runtime/pkg/controller/controllerutil" "sigs.k8s.io/controller-runtime/pkg/reconcile" kscontroller "kubesphere.io/kubesphere/pkg/controller" ) const ( controllerName = "workspace" finalizer = "finalizers.tenant.kubesphere.io" ) var _ kscontroller.Controller = &Reconciler{} var _ reconcile.Reconciler = &Reconciler{} // Reconciler reconciles a Workspace object type Reconciler struct { client.Client logger logr.Logger recorder record.EventRecorder } func (r *Reconciler) Name() string { return controllerName } func (r *Reconciler) SetupWithManager(mgr *kscontroller.Manager) error { r.Client = mgr.GetClient() r.logger = mgr.GetLogger().WithName(controllerName) r.recorder = mgr.GetEventRecorderFor(controllerName) return ctrl.NewControllerManagedBy(mgr). Named(controllerName). WithOptions(controller.Options{MaxConcurrentReconciles: 2}). For(&tenantv1beta1.Workspace{}). Complete(r) } // +kubebuilder:rbac:groups=tenant.kubesphere.io,resources=workspaces,verbs=get;list;watch;create;update;patch;delete // +kubebuilder:rbac:groups=tenant.kubesphere.io,resources=workspaces/status,verbs=get;update;patch // +kubebuilder:rbac:groups=iam.kubesphere.io,resources=users,verbs=get;list;watch // +kubebuilder:rbac:groups=iam.kubesphere.io,resources=rolebases,verbs=get;list;watch // +kubebuilder:rbac:groups=iam.kubesphere.io,resources=workspaceroles,verbs=get;list;watch;create;update;patch;delete // +kubebuilder:rbac:groups=iam.kubesphere.io,resources=workspacerolebindings,verbs=get;list;watch;create;update;patch;delete func (r *Reconciler) Reconcile(ctx context.Context, req ctrl.Request) (ctrl.Result, error) { logger := r.logger.WithValues("workspace", req.NamespacedName) ctx = klog.NewContext(ctx, logger) workspace := &tenantv1beta1.Workspace{} if err := r.Get(ctx, req.NamespacedName, workspace); err != nil { return ctrl.Result{}, client.IgnoreNotFound(err) } if workspace.ObjectMeta.DeletionTimestamp.IsZero() { // The object is not being deleted, so if it does not have our finalizer, // then lets add the finalizer and update the object. if !controllerutil.ContainsFinalizer(workspace, finalizer) { expected := workspace.DeepCopy() controllerutil.AddFinalizer(expected, finalizer) if err := r.Patch(ctx, expected, client.MergeFrom(workspace)); err != nil { return ctrl.Result{}, err } workspaceOperation.WithLabelValues("create", workspace.Name).Inc() } } else { // The object is being deleted if controllerutil.ContainsFinalizer(workspace, finalizer) { // remove our finalizer from the list and update it. controllerutil.RemoveFinalizer(workspace, finalizer) if err := r.Update(ctx, workspace); err != nil { return ctrl.Result{}, err } workspaceOperation.WithLabelValues("delete", workspace.Name).Inc() } // Our finalizer has finished, so the reconciler can do nothing. return ctrl.Result{}, nil } r.recorder.Event(workspace, corev1.EventTypeNormal, kscontroller.Synced, kscontroller.MessageResourceSynced) return ctrl.Result{}, nil }