From 6a5775ddce90b384f78cf6464a820fe1ec35d27e Mon Sep 17 00:00:00 2001 From: manju956 Date: Fri, 27 Sep 2024 13:26:05 +0530 Subject: [PATCH] RSCT operator status changes Signed-off-by: manju956 --- api/v1alpha1/rsct_types.go | 1 + api/v1alpha1/zz_generated.deepcopy.go | 7 +- .../rsct-operator.clusterserviceversion.yaml | 24 ++++-- bundle/manifests/rsct.ibm.com_rscts.yaml | 5 ++ config/crd/bases/rsct.ibm.com_rscts.yaml | 5 ++ .../rsct-operator.clusterserviceversion.yaml | 10 +-- config/rbac/role.yaml | 8 ++ internal/controller/rsct_controller.go | 1 + internal/controller/status.go | 82 +++++++++++++++++++ 9 files changed, 132 insertions(+), 11 deletions(-) diff --git a/api/v1alpha1/rsct_types.go b/api/v1alpha1/rsct_types.go index c20aead..10aa070 100644 --- a/api/v1alpha1/rsct_types.go +++ b/api/v1alpha1/rsct_types.go @@ -38,6 +38,7 @@ type RSCTSpec struct { type RSCTStatus struct { // INSERT ADDITIONAL STATUS FIELD - define observed state of cluster // Important: Run "make" to regenerate code after modifying this file + State *string `json:"state,omitempty"` } //+kubebuilder:object:root=true diff --git a/api/v1alpha1/zz_generated.deepcopy.go b/api/v1alpha1/zz_generated.deepcopy.go index bf0cd19..4af2d13 100644 --- a/api/v1alpha1/zz_generated.deepcopy.go +++ b/api/v1alpha1/zz_generated.deepcopy.go @@ -30,7 +30,7 @@ func (in *RSCT) DeepCopyInto(out *RSCT) { out.TypeMeta = in.TypeMeta in.ObjectMeta.DeepCopyInto(&out.ObjectMeta) in.Spec.DeepCopyInto(&out.Spec) - out.Status = in.Status + in.Status.DeepCopyInto(&out.Status) } // DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new RSCT. @@ -106,6 +106,11 @@ func (in *RSCTSpec) DeepCopy() *RSCTSpec { // DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil. func (in *RSCTStatus) DeepCopyInto(out *RSCTStatus) { *out = *in + if in.State != nil { + in, out := &in.State, &out.State + *out = new(string) + **out = **in + } } // DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new RSCTStatus. diff --git a/bundle/manifests/rsct-operator.clusterserviceversion.yaml b/bundle/manifests/rsct-operator.clusterserviceversion.yaml index a0ae0d2..25d87f7 100644 --- a/bundle/manifests/rsct-operator.clusterserviceversion.yaml +++ b/bundle/manifests/rsct-operator.clusterserviceversion.yaml @@ -17,15 +17,21 @@ metadata: }, "name": "rsct", "namespace": "rsct-operator-system" - } + }, + "spec": {} } ] capabilities: Basic Install - createdAt: "2024-07-10T12:35:54Z" + categories: OpenShift Optional + containerImage: ghcr.io/ocp-power-automation/rsct-operator:latest + createdAt: "2024-10-24T06:29:11Z" + description: Deploys RSCT daemonset on all nodes of an OpenShift cluster operators.operatorframework.io/builder: operator-sdk-v1.34.1 operators.operatorframework.io/project_layout: go.kubebuilder.io/v4 + repository: https://github.com/ocp-power-automation/rsct-operator + support: IBM name: rsct-operator.v0.0.1 - namespace: placeholder + namespace: rsct-operator-system spec: apiservicedefinitions: {} customresourcedefinitions: @@ -35,7 +41,7 @@ spec: kind: RSCT name: rscts.rsct.ibm.com version: v1alpha1 - description: Deploys RSCT on all nodes of an OpenShift cluster. + description: Deploys custom resource RSCT on all nodes of an OpenShift cluster. displayName: RSCT Operator for IBM Power Virtual Server icon: - base64data: "" @@ -44,6 +50,14 @@ spec: spec: clusterPermissions: - rules: + - apiGroups: + - "" + resources: + - pods + verbs: + - get + - list + - watch - apiGroups: - "" resources: @@ -255,7 +269,7 @@ spec: - power links: - name: Rsct Operator - url: https://rsct-operator.domain + url: https://github.com/ocp-power-automation/rsct-operator maintainers: - email: mjturek@us.ibm.com name: Michael Turek diff --git a/bundle/manifests/rsct.ibm.com_rscts.yaml b/bundle/manifests/rsct.ibm.com_rscts.yaml index 26d1b63..c15c7f8 100644 --- a/bundle/manifests/rsct.ibm.com_rscts.yaml +++ b/bundle/manifests/rsct.ibm.com_rscts.yaml @@ -46,6 +46,11 @@ spec: type: object status: description: RSCTStatus defines the observed state of RSCT + properties: + state: + description: |- + state reflects current observed state of RSCT resource + type: string type: object type: object served: true diff --git a/config/crd/bases/rsct.ibm.com_rscts.yaml b/config/crd/bases/rsct.ibm.com_rscts.yaml index 684e75f..711828d 100644 --- a/config/crd/bases/rsct.ibm.com_rscts.yaml +++ b/config/crd/bases/rsct.ibm.com_rscts.yaml @@ -46,6 +46,11 @@ spec: type: object status: description: RSCTStatus defines the observed state of RSCT + properties: + state: + description: |- + state reflects current observed state of RSCT resource + type: string type: object type: object served: true diff --git a/config/manifests/bases/rsct-operator.clusterserviceversion.yaml b/config/manifests/bases/rsct-operator.clusterserviceversion.yaml index 756bc6e..d7e4a4c 100644 --- a/config/manifests/bases/rsct-operator.clusterserviceversion.yaml +++ b/config/manifests/bases/rsct-operator.clusterserviceversion.yaml @@ -4,18 +4,18 @@ metadata: annotations: alm-examples: '[]' capabilities: Basic Install - support: IBM - categories: "OpenShift Optional" - containerImage: "ghcr.io/ocp-power-automation/rsct-operator:latest" - repository: "https://github.com/ocp-power-automation/rsct-operator" + categories: OpenShift Optional + containerImage: ghcr.io/ocp-power-automation/rsct-operator:latest description: Deploys RSCT daemonset on all nodes of an OpenShift cluster + repository: https://github.com/ocp-power-automation/rsct-operator + support: IBM name: rsct-operator.v0.0.0 namespace: rsct-operator-system spec: apiservicedefinitions: {} customresourcedefinitions: owned: - - description: RSCT custom resource is the schema for the rscts API + - description: RSCT is the Schema for the rscts API displayName: RSCT kind: RSCT name: rscts.rsct.ibm.com diff --git a/config/rbac/role.yaml b/config/rbac/role.yaml index 90bb3f6..e5d6eb4 100644 --- a/config/rbac/role.yaml +++ b/config/rbac/role.yaml @@ -4,6 +4,14 @@ kind: ClusterRole metadata: name: manager-role rules: +- apiGroups: + - "" + resources: + - pods + verbs: + - get + - list + - watch - apiGroups: - "" resources: diff --git a/internal/controller/rsct_controller.go b/internal/controller/rsct_controller.go index 1d01087..0bc4bed 100644 --- a/internal/controller/rsct_controller.go +++ b/internal/controller/rsct_controller.go @@ -51,6 +51,7 @@ type RSCTReconciler struct { //+kubebuilder:rbac:groups=rsct.ibm.com,resources=rscts/finalizers,verbs=update //+kubebuilder:rbac:groups=apps,resources=daemonsets,verbs=get;list;watch;create;update;patch;delete //+kubebuilder:rbac:groups="",resources=serviceaccounts,verbs=get;list;watch;create;update;patch;delete +//+kubebuilder:rbac:groups="",resources=pods,verbs=get;list;watch; // Reconcile is part of the main kubernetes reconciliation loop which aims to // move the current state of the cluster closer to the desired state. diff --git a/internal/controller/status.go b/internal/controller/status.go index 2b79cce..55a8125 100644 --- a/internal/controller/status.go +++ b/internal/controller/status.go @@ -2,11 +2,93 @@ package controller import ( "context" + "fmt" + "slices" rsctv1alpha1 "github.com/ocp-power-automation/rsct-operator/api/v1alpha1" appsv1 "k8s.io/api/apps/v1" + corev1 "k8s.io/api/core/v1" + "k8s.io/apimachinery/pkg/labels" + "sigs.k8s.io/controller-runtime/pkg/client" ) +type PodStatus string + +const ( + PENDING PodStatus = "PENDING" + RUNNING PodStatus = "RUNNING" + FAILED PodStatus = "FAILED" + PARTIALLY_RUNNING PodStatus = "PARTIALLY_RUNNING" + UNKNOWN PodStatus = "UNKNOWN" +) + +// matchPodsStatus checks if status of all the pods in slice are same or not +func matchPodsStatus(podsStatus []PodStatus, status PodStatus) bool { + for _, ps := range podsStatus { + if ps == status { + continue + } + return false + } + return true +} + +// evalOperatorStatus determines operator status based on the pods status +func evalOperatorStatus(podList *corev1.PodList) string { + var effectiveStatus PodStatus + var podsStatus []PodStatus + for _, pod := range podList.Items { + switch { + case pod.Status.Phase == corev1.PodPending: + podsStatus = append(podsStatus, PENDING) + case pod.Status.Phase == corev1.PodFailed: + podsStatus = append(podsStatus, FAILED) + case pod.Status.Phase == corev1.PodRunning: + podsStatus = append(podsStatus, RUNNING) + default: + podsStatus = append(podsStatus, UNKNOWN) + } + } + + if slices.Contains(podsStatus, RUNNING) { + if slices.Contains(podsStatus, FAILED) || slices.Contains(podsStatus, PENDING) || slices.Contains(podsStatus, UNKNOWN) { + effectiveStatus = PARTIALLY_RUNNING + } else { + effectiveStatus = RUNNING + } + } else if matchPodsStatus(podsStatus, FAILED) { + effectiveStatus = FAILED + } else if matchPodsStatus(podsStatus, PENDING) { + effectiveStatus = PENDING + } + return string(effectiveStatus) +} + +// updateRSCTStatus updates RSCT operator status func (r *RSCTReconciler) updateRSCTStatus(ctx context.Context, rsct *rsctv1alpha1.RSCT, currentDaemonSet *appsv1.DaemonSet) error { + // Operator status: + // 1. PENDING + // 2. RUNNING + // 3. PARTIALLY_RUNNING + // 4. FAILED + + pods := &corev1.PodList{} + + labelSelector := labels.SelectorFromSet(map[string]string{"app": currentDaemonSet.Name}) + listOpts := &client.ListOptions{Namespace: rsct.Namespace, LabelSelector: labelSelector} + listOpts.ApplyOptions([]client.ListOption{}) + + if err := r.List(ctx, pods, listOpts); err != nil { + return fmt.Errorf("failed to get list of rsct operator pods: %w", err) + } + + operatorStatus := evalOperatorStatus(pods) + rsct.Status.State = &operatorStatus + + err := r.Status().Update(ctx, rsct) + if err != nil { + return err + } + return nil }