mirror of
				https://github.com/optim-enterprises-bv/kubernetes.git
				synced 2025-11-03 19:58:17 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			164 lines
		
	
	
		
			4.8 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			164 lines
		
	
	
		
			4.8 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
/*
 | 
						|
Copyright 2015 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 (
 | 
						|
	"time"
 | 
						|
 | 
						|
	"k8s.io/kubernetes/pkg/api"
 | 
						|
	"k8s.io/kubernetes/pkg/api/meta"
 | 
						|
	"k8s.io/kubernetes/pkg/api/v1"
 | 
						|
	"k8s.io/kubernetes/pkg/client/restclient"
 | 
						|
	"k8s.io/kubernetes/pkg/fields"
 | 
						|
	"k8s.io/kubernetes/pkg/runtime"
 | 
						|
	"k8s.io/kubernetes/pkg/watch"
 | 
						|
)
 | 
						|
 | 
						|
// ListerWatcher is any object that knows how to perform an initial list and start a watch on a resource.
 | 
						|
type ListerWatcher interface {
 | 
						|
	// List should return a list type object; the Items field will be extracted, and the
 | 
						|
	// ResourceVersion field will be used to start the watch in the right place.
 | 
						|
	List(options v1.ListOptions) (runtime.Object, error)
 | 
						|
	// Watch should begin a watch at the specified version.
 | 
						|
	Watch(options v1.ListOptions) (watch.Interface, error)
 | 
						|
}
 | 
						|
 | 
						|
// ListFunc knows how to list resources
 | 
						|
type ListFunc func(options v1.ListOptions) (runtime.Object, error)
 | 
						|
 | 
						|
// WatchFunc knows how to watch resources
 | 
						|
type WatchFunc func(options v1.ListOptions) (watch.Interface, error)
 | 
						|
 | 
						|
// ListWatch knows how to list and watch a set of apiserver resources.  It satisfies the ListerWatcher interface.
 | 
						|
// It is a convenience function for users of NewReflector, etc.
 | 
						|
// ListFunc and WatchFunc must not be nil
 | 
						|
type ListWatch struct {
 | 
						|
	ListFunc  ListFunc
 | 
						|
	WatchFunc WatchFunc
 | 
						|
}
 | 
						|
 | 
						|
// Getter interface knows how to access Get method from RESTClient.
 | 
						|
type Getter interface {
 | 
						|
	Get() *restclient.Request
 | 
						|
}
 | 
						|
 | 
						|
// NewListWatchFromClient creates a new ListWatch from the specified client, resource, namespace and field selector.
 | 
						|
func NewListWatchFromClient(c Getter, resource string, namespace string, fieldSelector fields.Selector) *ListWatch {
 | 
						|
	listFunc := func(options v1.ListOptions) (runtime.Object, error) {
 | 
						|
		return c.Get().
 | 
						|
			Namespace(namespace).
 | 
						|
			Resource(resource).
 | 
						|
			VersionedParams(&options, api.ParameterCodec).
 | 
						|
			FieldsSelectorParam(fieldSelector).
 | 
						|
			Do().
 | 
						|
			Get()
 | 
						|
	}
 | 
						|
	watchFunc := func(options v1.ListOptions) (watch.Interface, error) {
 | 
						|
		return c.Get().
 | 
						|
			Prefix("watch").
 | 
						|
			Namespace(namespace).
 | 
						|
			Resource(resource).
 | 
						|
			VersionedParams(&options, api.ParameterCodec).
 | 
						|
			FieldsSelectorParam(fieldSelector).
 | 
						|
			Watch()
 | 
						|
	}
 | 
						|
	return &ListWatch{ListFunc: listFunc, WatchFunc: watchFunc}
 | 
						|
}
 | 
						|
 | 
						|
func timeoutFromListOptions(options v1.ListOptions) time.Duration {
 | 
						|
	if options.TimeoutSeconds != nil {
 | 
						|
		return time.Duration(*options.TimeoutSeconds) * time.Second
 | 
						|
	}
 | 
						|
	return 0
 | 
						|
}
 | 
						|
 | 
						|
// List a set of apiserver resources
 | 
						|
func (lw *ListWatch) List(options v1.ListOptions) (runtime.Object, error) {
 | 
						|
	return lw.ListFunc(options)
 | 
						|
}
 | 
						|
 | 
						|
// Watch a set of apiserver resources
 | 
						|
func (lw *ListWatch) Watch(options v1.ListOptions) (watch.Interface, error) {
 | 
						|
	return lw.WatchFunc(options)
 | 
						|
}
 | 
						|
 | 
						|
// TODO: check for watch expired error and retry watch from latest point?  Same issue exists for Until.
 | 
						|
func ListWatchUntil(timeout time.Duration, lw ListerWatcher, conditions ...watch.ConditionFunc) (*watch.Event, error) {
 | 
						|
	if len(conditions) == 0 {
 | 
						|
		return nil, nil
 | 
						|
	}
 | 
						|
 | 
						|
	list, err := lw.List(v1.ListOptions{})
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	initialItems, err := meta.ExtractList(list)
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
 | 
						|
	// use the initial items as simulated "adds"
 | 
						|
	var lastEvent *watch.Event
 | 
						|
	currIndex := 0
 | 
						|
	passedConditions := 0
 | 
						|
	for _, condition := range conditions {
 | 
						|
		// check the next condition against the previous event and short circuit waiting for the next watch
 | 
						|
		if lastEvent != nil {
 | 
						|
			done, err := condition(*lastEvent)
 | 
						|
			if err != nil {
 | 
						|
				return lastEvent, err
 | 
						|
			}
 | 
						|
			if done {
 | 
						|
				passedConditions = passedConditions + 1
 | 
						|
				continue
 | 
						|
			}
 | 
						|
		}
 | 
						|
 | 
						|
	ConditionSucceeded:
 | 
						|
		for currIndex < len(initialItems) {
 | 
						|
			lastEvent = &watch.Event{Type: watch.Added, Object: initialItems[currIndex]}
 | 
						|
			currIndex++
 | 
						|
 | 
						|
			done, err := condition(*lastEvent)
 | 
						|
			if err != nil {
 | 
						|
				return lastEvent, err
 | 
						|
			}
 | 
						|
			if done {
 | 
						|
				passedConditions = passedConditions + 1
 | 
						|
				break ConditionSucceeded
 | 
						|
			}
 | 
						|
		}
 | 
						|
	}
 | 
						|
	if passedConditions == len(conditions) {
 | 
						|
		return lastEvent, nil
 | 
						|
	}
 | 
						|
	remainingConditions := conditions[passedConditions:]
 | 
						|
 | 
						|
	metaObj, err := meta.ListAccessor(list)
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	currResourceVersion := metaObj.GetResourceVersion()
 | 
						|
 | 
						|
	watchInterface, err := lw.Watch(v1.ListOptions{ResourceVersion: currResourceVersion})
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
 | 
						|
	return watch.Until(timeout, watchInterface, remainingConditions...)
 | 
						|
}
 |