mirror of https://github.com/kubernetes/kops.git
				
				
				
			
		
			
				
	
	
		
			218 lines
		
	
	
		
			6.6 KiB
		
	
	
	
		
			Go
		
	
	
	
			
		
		
	
	
			218 lines
		
	
	
		
			6.6 KiB
		
	
	
	
		
			Go
		
	
	
	
| /*
 | |
| 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"
 | |
| 	"reflect"
 | |
| 	"strings"
 | |
| 
 | |
| 	apimeta "k8s.io/apimachinery/pkg/api/meta"
 | |
| 	"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"
 | |
| }
 | |
| 
 | |
| // informerCache is a Kubernetes Object cache populated from InformersMap.  informerCache wraps an InformersMap.
 | |
| type informerCache struct {
 | |
| 	*internal.InformersMap
 | |
| }
 | |
| 
 | |
| // Get implements Reader.
 | |
| func (ip *informerCache) Get(ctx context.Context, key client.ObjectKey, out client.Object) error {
 | |
| 	gvk, err := apiutil.GVKForObject(out, ip.Scheme)
 | |
| 	if err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 
 | |
| 	started, cache, err := ip.InformersMap.Get(ctx, gvk, out)
 | |
| 	if err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 
 | |
| 	if !started {
 | |
| 		return &ErrCacheNotStarted{}
 | |
| 	}
 | |
| 	return cache.Reader.Get(ctx, key, out)
 | |
| }
 | |
| 
 | |
| // List implements Reader.
 | |
| func (ip *informerCache) List(ctx context.Context, out client.ObjectList, opts ...client.ListOption) error {
 | |
| 	gvk, cacheTypeObj, err := ip.objectTypeForListObject(out)
 | |
| 	if err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 
 | |
| 	started, cache, err := ip.InformersMap.Get(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 (ip *informerCache) objectTypeForListObject(list client.ObjectList) (*schema.GroupVersionKind, runtime.Object, error) {
 | |
| 	gvk, err := apiutil.GVKForObject(list, ip.Scheme)
 | |
| 	if err != nil {
 | |
| 		return nil, nil, err
 | |
| 	}
 | |
| 
 | |
| 	if !strings.HasSuffix(gvk.Kind, "List") {
 | |
| 		return nil, nil, fmt.Errorf("non-list type %T (kind %q) passed as output", list, gvk)
 | |
| 	}
 | |
| 	// we need the non-list GVK, so chop off the "List" from the end of the kind
 | |
| 	gvk.Kind = gvk.Kind[:len(gvk.Kind)-4]
 | |
| 	_, isUnstructured := list.(*unstructured.UnstructuredList)
 | |
| 	var cacheTypeObj runtime.Object
 | |
| 	if isUnstructured {
 | |
| 		u := &unstructured.Unstructured{}
 | |
| 		u.SetGroupVersionKind(gvk)
 | |
| 		cacheTypeObj = u
 | |
| 	} else {
 | |
| 		itemsPtr, err := apimeta.GetItemsPtr(list)
 | |
| 		if err != nil {
 | |
| 			return nil, nil, err
 | |
| 		}
 | |
| 		// http://knowyourmeme.com/memes/this-is-fine
 | |
| 		elemType := reflect.Indirect(reflect.ValueOf(itemsPtr)).Type().Elem()
 | |
| 		if elemType.Kind() != reflect.Ptr {
 | |
| 			elemType = reflect.PtrTo(elemType)
 | |
| 		}
 | |
| 
 | |
| 		cacheTypeValue := reflect.Zero(elemType)
 | |
| 		var ok bool
 | |
| 		cacheTypeObj, ok = cacheTypeValue.Interface().(runtime.Object)
 | |
| 		if !ok {
 | |
| 			return nil, nil, fmt.Errorf("cannot get cache for %T, its element %T is not a runtime.Object", list, cacheTypeValue.Interface())
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return &gvk, cacheTypeObj, nil
 | |
| }
 | |
| 
 | |
| // GetInformerForKind returns the informer for the GroupVersionKind.
 | |
| func (ip *informerCache) GetInformerForKind(ctx context.Context, gvk schema.GroupVersionKind) (Informer, error) {
 | |
| 	// Map the gvk to an object
 | |
| 	obj, err := ip.Scheme.New(gvk)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 
 | |
| 	_, i, err := ip.InformersMap.Get(ctx, gvk, obj)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 	return i.Informer, err
 | |
| }
 | |
| 
 | |
| // GetInformer returns the informer for the obj.
 | |
| func (ip *informerCache) GetInformer(ctx context.Context, obj client.Object) (Informer, error) {
 | |
| 	gvk, err := apiutil.GVKForObject(obj, ip.Scheme)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 
 | |
| 	_, i, err := ip.InformersMap.Get(ctx, gvk, obj)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 	return i.Informer, err
 | |
| }
 | |
| 
 | |
| // NeedLeaderElection implements the LeaderElectionRunnable interface
 | |
| // to indicate that this can be started without requiring the leader lock.
 | |
| func (ip *informerCache) NeedLeaderElection() bool {
 | |
| 	return false
 | |
| }
 | |
| 
 | |
| // IndexField adds an indexer to the underlying cache, using extraction 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 (ip *informerCache) IndexField(ctx context.Context, obj client.Object, field string, extractValue client.IndexerFunc) error {
 | |
| 	informer, err := ip.GetInformer(ctx, obj)
 | |
| 	if err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 	return indexByField(informer, field, extractValue)
 | |
| }
 | |
| 
 | |
| func indexByField(indexer Informer, field string, extractor 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 := extractor(obj)
 | |
| 		var vals []string
 | |
| 		if ns == "" {
 | |
| 			// if we're not doubling the keys for the namespaced case, just re-use what was returned to us
 | |
| 			vals = 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 indexer.AddIndexers(cache.Indexers{internal.FieldIndexName(field): indexFunc})
 | |
| }
 |