/* Copyright 2024. 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 helper import ( "reflect" appsv1 "k8s.io/api/apps/v1" corev1 "k8s.io/api/core/v1" "k8s.io/apimachinery/pkg/api/resource" kubefloworgv1beta1 "github.com/kubeflow/notebooks/workspaces/controller/api/v1beta1" ) // CopyStatefulSetFields updates a target StatefulSet with the fields from a desired StatefulSet, returning true if an update is required. func CopyStatefulSetFields(desired *appsv1.StatefulSet, target *appsv1.StatefulSet) bool { requireUpdate := false // copy `metadata.labels` for k, v := range target.Labels { if desired.Labels[k] != v { requireUpdate = true } } target.Labels = desired.Labels // copy `metadata.annotations` for k, v := range target.Annotations { if desired.Annotations[k] != v { requireUpdate = true } } target.Annotations = desired.Annotations // copy `spec.replicas` if *desired.Spec.Replicas != *target.Spec.Replicas { *target.Spec.Replicas = *desired.Spec.Replicas requireUpdate = true } // copy `spec.selector` // // TODO: confirm if StatefulSets support updates to the selector // if not, we might need to recreate the StatefulSet // if !reflect.DeepEqual(target.Spec.Selector, desired.Spec.Selector) { target.Spec.Selector = desired.Spec.Selector requireUpdate = true } // copy `spec.template` // // TODO: confirm if there is a problem with doing the update at the `spec.template` level // or if only `spec.template.spec` should be updated // if !reflect.DeepEqual(target.Spec.Template, desired.Spec.Template) { target.Spec.Template = desired.Spec.Template requireUpdate = true } return requireUpdate } // CopyServiceFields updates a target Service with the fields from a desired Service, returning true if an update is required. func CopyServiceFields(desired *corev1.Service, target *corev1.Service) bool { requireUpdate := false // copy `metadata.labels` for k, v := range target.Labels { if desired.Labels[k] != v { requireUpdate = true } } target.Labels = desired.Labels // copy `metadata.annotations` for k, v := range target.Annotations { if desired.Annotations[k] != v { requireUpdate = true } } target.Annotations = desired.Annotations // NOTE: we don't copy the entire `spec` because we can't overwrite the `spec.clusterIp` and similar fields // copy `spec.ports` if !reflect.DeepEqual(target.Spec.Ports, desired.Spec.Ports) { target.Spec.Ports = desired.Spec.Ports requireUpdate = true } // copy `spec.selector` if !reflect.DeepEqual(target.Spec.Selector, desired.Spec.Selector) { target.Spec.Selector = desired.Spec.Selector requireUpdate = true } // copy `spec.type` if target.Spec.Type != desired.Spec.Type { target.Spec.Type = desired.Spec.Type requireUpdate = true } return requireUpdate } // NormalizePodConfigSpec normalizes a PodConfigSpec so that it can be compared with reflect.DeepEqual func NormalizePodConfigSpec(spec kubefloworgv1beta1.PodConfigSpec) error { // normalize Affinity if spec.Affinity != nil { // set Affinity to nil if it is empty if reflect.DeepEqual(spec.Affinity, corev1.Affinity{}) { spec.Affinity = nil } } // normalize NodeSelector if spec.NodeSelector != nil { // set NodeSelector to nil if it is empty if len(spec.NodeSelector) == 0 { spec.NodeSelector = nil } } // normalize Tolerations if spec.Tolerations != nil { // set Tolerations to nil if it is empty if len(spec.Tolerations) == 0 { spec.Tolerations = nil } } // normalize Resources if spec.Resources != nil { // if Resources.Requests is empty, set it to nil if len(spec.Resources.Requests) == 0 { spec.Resources.Requests = nil } else { // otherwise, normalize the values in Resources.Requests for key, value := range spec.Resources.Requests { q, err := resource.ParseQuantity(value.String()) if err != nil { return err } spec.Resources.Requests[key] = q } } // if Resources.Limits is empty, set it to nil if len(spec.Resources.Limits) == 0 { spec.Resources.Limits = nil } else { // otherwise, normalize the values in Resources.Limits for key, value := range spec.Resources.Limits { q, err := resource.ParseQuantity(value.String()) if err != nil { return err } spec.Resources.Limits[key] = q } } } return nil }