/* Copyright 2018 The Kubernetes Authors. 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 cache import ( "context" "fmt" "strings" apimeta "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/client-go/tools/cache" "sigs.k8s.io/controller-runtime/pkg/cache/internal" "sigs.k8s.io/controller-runtime/pkg/client" "sigs.k8s.io/controller-runtime/pkg/client/apiutil" ) var ( _ Informers = &informerCache{} _ client.Reader = &informerCache{} _ Cache = &informerCache{} ) // ErrCacheNotStarted is returned when trying to read from the cache that wasn't started. type ErrCacheNotStarted struct{} func (*ErrCacheNotStarted) Error() string { return "the cache is not started, can not read objects" } var _ error = (*ErrCacheNotStarted)(nil) // ErrResourceNotCached indicates that the resource type // the client asked the cache for is not cached, i.e. the // corresponding informer does not exist yet. type ErrResourceNotCached struct { GVK schema.GroupVersionKind } // Error returns the error func (r ErrResourceNotCached) Error() string { return fmt.Sprintf("%s is not cached", r.GVK.String()) } var _ error = (*ErrResourceNotCached)(nil) // informerCache is a Kubernetes Object cache populated from internal.Informers. // informerCache wraps internal.Informers. type informerCache struct { scheme *runtime.Scheme *internal.Informers readerFailOnMissingInformer bool } // Get implements Reader. func (ic *informerCache) Get(ctx context.Context, key client.ObjectKey, out client.Object, opts ...client.GetOption) error { gvk, err := apiutil.GVKForObject(out, ic.scheme) if err != nil { return err } started, cache, err := ic.getInformerForKind(ctx, gvk, out) if err != nil { return err } if !started { return &ErrCacheNotStarted{} } return cache.Reader.Get(ctx, key, out, opts...) } // List implements Reader. func (ic *informerCache) List(ctx context.Context, out client.ObjectList, opts ...client.ListOption) error { gvk, cacheTypeObj, err := ic.objectTypeForListObject(out) if err != nil { return err } started, cache, err := ic.getInformerForKind(ctx, *gvk, cacheTypeObj) if err != nil { return err } if !started { return &ErrCacheNotStarted{} } return cache.Reader.List(ctx, out, opts...) } // objectTypeForListObject tries to find the runtime.Object and associated GVK // for a single object corresponding to the passed-in list type. We need them // because they are used as cache map key. func (ic *informerCache) objectTypeForListObject(list client.ObjectList) (*schema.GroupVersionKind, runtime.Object, error) { gvk, err := apiutil.GVKForObject(list, ic.scheme) if err != nil { return nil, nil, err } // We need the non-list GVK, so chop off the "List" from the end of the kind. gvk.Kind = strings.TrimSuffix(gvk.Kind, "List") // Handle unstructured.UnstructuredList. if _, isUnstructured := list.(runtime.Unstructured); isUnstructured { u := &unstructured.Unstructured{} u.SetGroupVersionKind(gvk) return &gvk, u, nil } // Handle metav1.PartialObjectMetadataList. if _, isPartialObjectMetadata := list.(*metav1.PartialObjectMetadataList); isPartialObjectMetadata { pom := &metav1.PartialObjectMetadata{} pom.SetGroupVersionKind(gvk) return &gvk, pom, nil } // Any other list type should have a corresponding non-list type registered // in the scheme. Use that to create a new instance of the non-list type. cacheTypeObj, err := ic.scheme.New(gvk) if err != nil { return nil, nil, err } return &gvk, cacheTypeObj, nil } func applyGetOptions(opts ...InformerGetOption) *internal.GetOptions { cfg := &InformerGetOptions{} for _, opt := range opts { opt(cfg) } return (*internal.GetOptions)(cfg) } // GetInformerForKind returns the informer for the GroupVersionKind. If no informer exists, one will be started. func (ic *informerCache) GetInformerForKind(ctx context.Context, gvk schema.GroupVersionKind, opts ...InformerGetOption) (Informer, error) { // Map the gvk to an object obj, err := ic.scheme.New(gvk) if err != nil { return nil, err } _, i, err := ic.Informers.Get(ctx, gvk, obj, applyGetOptions(opts...)) if err != nil { return nil, err } return i.Informer, nil } // GetInformer returns the informer for the obj. If no informer exists, one will be started. func (ic *informerCache) GetInformer(ctx context.Context, obj client.Object, opts ...InformerGetOption) (Informer, error) { gvk, err := apiutil.GVKForObject(obj, ic.scheme) if err != nil { return nil, err } _, i, err := ic.Informers.Get(ctx, gvk, obj, applyGetOptions(opts...)) if err != nil { return nil, err } return i.Informer, nil } func (ic *informerCache) getInformerForKind(ctx context.Context, gvk schema.GroupVersionKind, obj runtime.Object) (bool, *internal.Cache, error) { if ic.readerFailOnMissingInformer { cache, started, ok := ic.Informers.Peek(gvk, obj) if !ok { return false, nil, &ErrResourceNotCached{GVK: gvk} } return started, cache, nil } return ic.Informers.Get(ctx, gvk, obj, &internal.GetOptions{}) } // NeedLeaderElection implements the LeaderElectionRunnable interface // to indicate that this can be started without requiring the leader lock. func (ic *informerCache) NeedLeaderElection() bool { return false } // IndexField adds an indexer to the underlying informer, using extractValue function to get // value(s) from the given field. This index can then be used by passing a field selector // to List. For one-to-one compatibility with "normal" field selectors, only return one value. // The values may be anything. They will automatically be prefixed with the namespace of the // given object, if present. The objects passed are guaranteed to be objects of the correct type. func (ic *informerCache) IndexField(ctx context.Context, obj client.Object, field string, extractValue client.IndexerFunc) error { informer, err := ic.GetInformer(ctx, obj) if err != nil { return err } return indexByField(informer, field, extractValue) } func indexByField(informer Informer, field string, extractValue client.IndexerFunc) error { indexFunc := func(objRaw interface{}) ([]string, error) { // TODO(directxman12): check if this is the correct type? obj, isObj := objRaw.(client.Object) if !isObj { return nil, fmt.Errorf("object of type %T is not an Object", objRaw) } meta, err := apimeta.Accessor(obj) if err != nil { return nil, err } ns := meta.GetNamespace() rawVals := extractValue(obj) var vals []string if ns == "" { // if we're not doubling the keys for the namespaced case, just create a new slice with same length vals = make([]string, len(rawVals)) } else { // if we need to add non-namespaced versions too, double the length vals = make([]string, len(rawVals)*2) } for i, rawVal := range rawVals { // save a namespaced variant, so that we can ask // "what are all the object matching a given index *in a given namespace*" vals[i] = internal.KeyToNamespacedKey(ns, rawVal) if ns != "" { // if we have a namespace, also inject a special index key for listing // regardless of the object namespace vals[i+len(rawVals)] = internal.KeyToNamespacedKey("", rawVal) } } return vals, nil } return informer.AddIndexers(cache.Indexers{internal.FieldIndexName(field): indexFunc}) }