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..d86f9a7 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: @@ -173,7 +187,7 @@ spec: - --leader-elect command: - /manager - image: ghcr.io/ocp-power-automation/rsct-operator:latest + image: ghcr.io/ocp-power-automation/rsct-operator:0.0.1 livenessProbe: httpGet: path: /healthz @@ -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..41c252a 100644 --- a/bundle/manifests/rsct.ibm.com_rscts.yaml +++ b/bundle/manifests/rsct.ibm.com_rscts.yaml @@ -46,6 +46,12 @@ spec: type: object status: description: RSCTStatus defines the observed state of RSCT + properties: + state: + description: |- + INSERT ADDITIONAL STATUS FIELD - define observed state of cluster + Important: Run "make" to regenerate code after modifying this file + 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..b70b14b 100644 --- a/config/crd/bases/rsct.ibm.com_rscts.yaml +++ b/config/crd/bases/rsct.ibm.com_rscts.yaml @@ -46,6 +46,12 @@ spec: type: object status: description: RSCTStatus defines the observed state of RSCT + properties: + state: + description: |- + INSERT ADDITIONAL STATUS FIELD - define observed state of cluster + Important: Run "make" to regenerate code after modifying this file + type: string type: object type: object served: true diff --git a/config/manager/kustomization.yaml b/config/manager/kustomization.yaml index 61f1d41..a69a518 100644 --- a/config/manager/kustomization.yaml +++ b/config/manager/kustomization.yaml @@ -13,4 +13,4 @@ kind: Kustomization images: - name: controller newName: ghcr.io/ocp-power-automation/rsct-operator - newTag: latest + newTag: 0.0.1 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/go.mod b/go.mod index 24a0225..40baff1 100644 --- a/go.mod +++ b/go.mod @@ -16,6 +16,7 @@ require ( require ( github.com/beorn7/perks v1.0.1 // indirect + github.com/cenkalti/backoff v2.2.1+incompatible // indirect github.com/cespare/xxhash/v2 v2.3.0 // indirect github.com/davecgh/go-spew v1.1.1 // indirect github.com/emicklei/go-restful/v3 v3.12.0 // indirect diff --git a/go.sum b/go.sum index ede4e26..e29278d 100644 --- a/go.sum +++ b/go.sum @@ -1,5 +1,7 @@ github.com/beorn7/perks v1.0.1 h1:VlbKKnNfV8bJzeqoa4cOKqO6bYr3WgKZxO8Z16+hsOM= github.com/beorn7/perks v1.0.1/go.mod h1:G2ZrVWU2WbWT9wwq4/hrbKbnv/1ERSJQ0ibhJ6rlkpw= +github.com/cenkalti/backoff v2.2.1+incompatible h1:tNowT99t7UNflLxfYYSlKYsBpXdEet03Pg2g16Swow4= +github.com/cenkalti/backoff v2.2.1+incompatible/go.mod h1:90ReRw6GdpyfrHakVjL/QHaoyV4aDUVVkXQJJJ3NXXM= github.com/cespare/xxhash/v2 v2.3.0 h1:UL815xU9SqsFlibzuggzjXhog7bL6oX9BbNZnL2UFvs= github.com/cespare/xxhash/v2 v2.3.0/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs= github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= 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..cc4252e 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 = "RUNNING" + 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 []string, status string) 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 string + var podsStatus []string + for _, pod := range podList.Items { + switch { + case pod.Status.Phase == corev1.PodPending: + podsStatus = append(podsStatus, string(PENDING)) + case pod.Status.Phase == corev1.PodFailed: + podsStatus = append(podsStatus, string(FAILED)) + case pod.Status.Phase == corev1.PodRunning: + podsStatus = append(podsStatus, string(RUNNING)) + default: + podsStatus = append(podsStatus, string(UNKNOWN)) + } + } + + if slices.Contains(podsStatus, string(RUNNING)) { + if slices.Contains(podsStatus, string(FAILED)) || slices.Contains(podsStatus, string(PENDING)) || slices.Contains(podsStatus, string(UNKNOWN)) { + effectiveStatus = string(PARTIALLY_RUNNING) + } else { + effectiveStatus = string(RUNNING) + } + } else if matchPodsStatus(podsStatus, string(FAILED)) { + effectiveStatus = string(FAILED) + } else if matchPodsStatus(podsStatus, string(PENDING)) { + effectiveStatus = string(PENDING) + } + return 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 }