karmada/pkg/util/helper/cache.go

78 lines
3.0 KiB
Go

package helper
import (
"context"
"fmt"
apierrors "k8s.io/apimachinery/pkg/api/errors"
"k8s.io/apimachinery/pkg/api/meta"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
"k8s.io/apimachinery/pkg/apis/meta/v1/unstructured"
"k8s.io/apimachinery/pkg/runtime"
"k8s.io/apimachinery/pkg/runtime/schema"
"k8s.io/klog/v2"
"sigs.k8s.io/controller-runtime/pkg/client"
"github.com/karmada-io/karmada/pkg/util"
"github.com/karmada-io/karmada/pkg/util/informermanager"
"github.com/karmada-io/karmada/pkg/util/informermanager/keys"
"github.com/karmada-io/karmada/pkg/util/restmapper"
)
type clusterDynamicClientSetFunc func(clusterName string, client client.Client) (*util.DynamicClusterClient, error)
// GetObjectFromCache gets full object information from cache by key in worker queue.
func GetObjectFromCache(restMapper meta.RESTMapper, manager informermanager.MultiClusterInformerManager, fedKey keys.FederatedKey,
client client.Client, clientSetFunc clusterDynamicClientSetFunc) (*unstructured.Unstructured, error) {
gvr, err := restmapper.GetGroupVersionResource(restMapper, fedKey.GroupVersionKind())
if err != nil {
klog.Errorf("Failed to get GVR from GVK %s. Error: %v", fedKey.GroupVersionKind(), err)
return nil, err
}
singleClusterManager := manager.GetSingleClusterManager(fedKey.Cluster)
if singleClusterManager == nil {
// That may happen in case of multi-controllers sharing one informer. For example:
// controller-A takes responsibility of initialize informer for clusters, but controller-B consumes
// the informer before the initialization.
// Usually this error will be eliminated during the controller reconciling loop.
return nil, fmt.Errorf("the informer of cluster(%s) has not been initialized", fedKey.Cluster)
}
if !singleClusterManager.IsInformerSynced(gvr) {
// fall back to call api server in case the cache has not been synchronized yet
return getObjectFromMemberCluster(gvr, fedKey, client, clientSetFunc)
}
var obj runtime.Object
lister := singleClusterManager.Lister(gvr)
obj, err = lister.Get(fedKey.NamespaceKey())
if err != nil {
if apierrors.IsNotFound(err) {
return nil, err
}
// print logs only for real error.
klog.Errorf("Failed to get obj %s. error: %v.", fedKey.String(), err)
return nil, err
}
return obj.(*unstructured.Unstructured), nil
}
// getObjectFromMemberCluster will try to get resource from member cluster by DynamicClientSet.
func getObjectFromMemberCluster(gvr schema.GroupVersionResource, fedKey keys.FederatedKey, client client.Client,
clientSetFunc clusterDynamicClientSetFunc) (*unstructured.Unstructured, error) {
dynamicClusterClient, err := clientSetFunc(fedKey.Cluster, client)
if err != nil {
klog.Errorf("Failed to build dynamic cluster client for cluster %s.", fedKey.Cluster)
return nil, err
}
existObj, err := dynamicClusterClient.DynamicClientSet.Resource(gvr).Namespace(fedKey.Namespace).Get(context.TODO(),
fedKey.Name, metav1.GetOptions{})
if err != nil {
return nil, err
}
return existObj, nil
}