mirror of
https://github.com/kubernetes/client-go.git
synced 2025-06-22 13:17:07 +00:00
The context is used for cancellation and to support contextual logging. In most cases, alternative *WithContext APIs get added, except for NewIntegerResourceVersionMutationCache where code searches indicate that the API is not used downstream. An API break around SharedInformer couldn't be avoided because the alternative (keeping the interface unchanged and adding a second one with the new method) would have been worse. controller-runtime needs to be updated because it implements that interface in a test package. Downstream consumers of controller-runtime will work unless they use those test package. Converting Kubernetes to use the other new alternatives will follow. In the meantime, usage of the new alternatives cannot be enforced via logcheck yet (see https://github.com/kubernetes/kubernetes/issues/126379 for the process). Passing context through and checking it for cancellation is tricky for event handlers. A better approach is to map the context cancellation to the normal removal of an event handler via a helper goroutine. Thanks to the new HandleErrorWithLogr and HandleCrashWithLogr, remembering the logger is sufficient for handling problems at runtime. Kubernetes-commit: 4638ba971661497b147906b8977ae206c9dd6e44
177 lines
5.2 KiB
Go
177 lines
5.2 KiB
Go
/*
|
|
Copyright 2014 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 (
|
|
"k8s.io/klog/v2"
|
|
|
|
"k8s.io/apimachinery/pkg/api/errors"
|
|
"k8s.io/apimachinery/pkg/api/meta"
|
|
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
|
"k8s.io/apimachinery/pkg/labels"
|
|
"k8s.io/apimachinery/pkg/runtime"
|
|
"k8s.io/apimachinery/pkg/runtime/schema"
|
|
)
|
|
|
|
// AppendFunc is used to add a matching item to whatever list the caller is using
|
|
type AppendFunc func(interface{})
|
|
|
|
// ListAll lists items in the store matching the given selector, calling appendFn on each one.
|
|
func ListAll(store Store, selector labels.Selector, appendFn AppendFunc) error {
|
|
selectAll := selector.Empty()
|
|
for _, m := range store.List() {
|
|
if selectAll {
|
|
// Avoid computing labels of the objects to speed up common flows
|
|
// of listing all objects.
|
|
appendFn(m)
|
|
continue
|
|
}
|
|
metadata, err := meta.Accessor(m)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if selector.Matches(labels.Set(metadata.GetLabels())) {
|
|
appendFn(m)
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// ListAllByNamespace lists items in the given namespace in the store matching the given selector,
|
|
// calling appendFn on each one.
|
|
// If a blank namespace (NamespaceAll) is specified, this delegates to ListAll().
|
|
func ListAllByNamespace(indexer Indexer, namespace string, selector labels.Selector, appendFn AppendFunc) error {
|
|
if namespace == metav1.NamespaceAll {
|
|
return ListAll(indexer, selector, appendFn)
|
|
}
|
|
|
|
items, err := indexer.Index(NamespaceIndex, &metav1.ObjectMeta{Namespace: namespace})
|
|
if err != nil {
|
|
// Ignore error; do slow search without index.
|
|
//
|
|
// ListAllByNamespace is called by generated code
|
|
// (k8s.io/client-go/listers) and probably not worth converting
|
|
// to contextual logging, which would require changing all of
|
|
// those APIs.
|
|
klog.TODO().Info("Warning: can not retrieve list of objects using index", "err", err)
|
|
for _, m := range indexer.List() {
|
|
metadata, err := meta.Accessor(m)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if metadata.GetNamespace() == namespace && selector.Matches(labels.Set(metadata.GetLabels())) {
|
|
appendFn(m)
|
|
}
|
|
|
|
}
|
|
return nil
|
|
}
|
|
|
|
selectAll := selector.Empty()
|
|
for _, m := range items {
|
|
if selectAll {
|
|
// Avoid computing labels of the objects to speed up common flows
|
|
// of listing all objects.
|
|
appendFn(m)
|
|
continue
|
|
}
|
|
metadata, err := meta.Accessor(m)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if selector.Matches(labels.Set(metadata.GetLabels())) {
|
|
appendFn(m)
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// GenericLister is a lister skin on a generic Indexer
|
|
type GenericLister interface {
|
|
// List will return all objects across namespaces
|
|
List(selector labels.Selector) (ret []runtime.Object, err error)
|
|
// Get will attempt to retrieve assuming that name==key
|
|
Get(name string) (runtime.Object, error)
|
|
// ByNamespace will give you a GenericNamespaceLister for one namespace
|
|
ByNamespace(namespace string) GenericNamespaceLister
|
|
}
|
|
|
|
// GenericNamespaceLister is a lister skin on a generic Indexer
|
|
type GenericNamespaceLister interface {
|
|
// List will return all objects in this namespace
|
|
List(selector labels.Selector) (ret []runtime.Object, err error)
|
|
// Get will attempt to retrieve by namespace and name
|
|
Get(name string) (runtime.Object, error)
|
|
}
|
|
|
|
// NewGenericLister creates a new instance for the genericLister.
|
|
func NewGenericLister(indexer Indexer, resource schema.GroupResource) GenericLister {
|
|
return &genericLister{indexer: indexer, resource: resource}
|
|
}
|
|
|
|
type genericLister struct {
|
|
indexer Indexer
|
|
resource schema.GroupResource
|
|
}
|
|
|
|
func (s *genericLister) List(selector labels.Selector) (ret []runtime.Object, err error) {
|
|
err = ListAll(s.indexer, selector, func(m interface{}) {
|
|
ret = append(ret, m.(runtime.Object))
|
|
})
|
|
return ret, err
|
|
}
|
|
|
|
func (s *genericLister) ByNamespace(namespace string) GenericNamespaceLister {
|
|
return &genericNamespaceLister{indexer: s.indexer, namespace: namespace, resource: s.resource}
|
|
}
|
|
|
|
func (s *genericLister) Get(name string) (runtime.Object, error) {
|
|
obj, exists, err := s.indexer.GetByKey(name)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if !exists {
|
|
return nil, errors.NewNotFound(s.resource, name)
|
|
}
|
|
return obj.(runtime.Object), nil
|
|
}
|
|
|
|
type genericNamespaceLister struct {
|
|
indexer Indexer
|
|
namespace string
|
|
resource schema.GroupResource
|
|
}
|
|
|
|
func (s *genericNamespaceLister) List(selector labels.Selector) (ret []runtime.Object, err error) {
|
|
err = ListAllByNamespace(s.indexer, s.namespace, selector, func(m interface{}) {
|
|
ret = append(ret, m.(runtime.Object))
|
|
})
|
|
return ret, err
|
|
}
|
|
|
|
func (s *genericNamespaceLister) Get(name string) (runtime.Object, error) {
|
|
obj, exists, err := s.indexer.GetByKey(s.namespace + "/" + name)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if !exists {
|
|
return nil, errors.NewNotFound(s.resource, name)
|
|
}
|
|
return obj.(runtime.Object), nil
|
|
}
|