mirror of
				https://github.com/k3s-io/kubernetes.git
				synced 2025-10-26 11:07:45 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			536 lines
		
	
	
		
			18 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			536 lines
		
	
	
		
			18 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.
 | |
| */
 | |
| 
 | |
| // CAUTION: If you update code in this file, you may need to also update code
 | |
| //          in contrib/mesos/pkg/service/endpoints_controller.go
 | |
| package endpoint
 | |
| 
 | |
| import (
 | |
| 	"reflect"
 | |
| 	"strconv"
 | |
| 	"time"
 | |
| 
 | |
| 	"encoding/json"
 | |
| 
 | |
| 	"github.com/golang/glog"
 | |
| 	"k8s.io/kubernetes/pkg/api"
 | |
| 	"k8s.io/kubernetes/pkg/api/endpoints"
 | |
| 	"k8s.io/kubernetes/pkg/api/errors"
 | |
| 	podutil "k8s.io/kubernetes/pkg/api/pod"
 | |
| 	utilpod "k8s.io/kubernetes/pkg/api/pod"
 | |
| 	"k8s.io/kubernetes/pkg/client/cache"
 | |
| 	clientset "k8s.io/kubernetes/pkg/client/clientset_generated/internalclientset"
 | |
| 	"k8s.io/kubernetes/pkg/controller"
 | |
| 	"k8s.io/kubernetes/pkg/controller/framework"
 | |
| 	"k8s.io/kubernetes/pkg/controller/framework/informers"
 | |
| 	"k8s.io/kubernetes/pkg/labels"
 | |
| 	"k8s.io/kubernetes/pkg/runtime"
 | |
| 	"k8s.io/kubernetes/pkg/util/metrics"
 | |
| 	utilruntime "k8s.io/kubernetes/pkg/util/runtime"
 | |
| 	"k8s.io/kubernetes/pkg/util/sets"
 | |
| 	"k8s.io/kubernetes/pkg/util/wait"
 | |
| 	"k8s.io/kubernetes/pkg/util/workqueue"
 | |
| 	"k8s.io/kubernetes/pkg/watch"
 | |
| )
 | |
| 
 | |
| const (
 | |
| 	// We'll attempt to recompute EVERY service's endpoints at least this
 | |
| 	// often. Higher numbers = lower CPU/network load; lower numbers =
 | |
| 	// shorter amount of time before a mistaken endpoint is corrected.
 | |
| 	FullServiceResyncPeriod = 30 * time.Second
 | |
| 
 | |
| 	// We must avoid syncing service until the pod store has synced. If it hasn't synced, to
 | |
| 	// avoid a hot loop, we'll wait this long between checks.
 | |
| 	PodStoreSyncedPollPeriod = 100 * time.Millisecond
 | |
| 
 | |
| 	// An annotation on the Service denoting if the endpoints controller should
 | |
| 	// go ahead and create endpoints for unready pods. This annotation is
 | |
| 	// currently only used by PetSets, where we need the pet to be DNS
 | |
| 	// resolvable during initialization. In this situation we create a headless
 | |
| 	// service just for the PetSet, and clients shouldn't be using this Service
 | |
| 	// for anything so unready endpoints don't matter.
 | |
| 	TolerateUnreadyEndpointsAnnotation = "service.alpha.kubernetes.io/tolerate-unready-endpoints"
 | |
| )
 | |
| 
 | |
| var (
 | |
| 	keyFunc = framework.DeletionHandlingMetaNamespaceKeyFunc
 | |
| )
 | |
| 
 | |
| // NewEndpointController returns a new *EndpointController.
 | |
| func NewEndpointController(podInformer framework.SharedIndexInformer, client *clientset.Clientset) *EndpointController {
 | |
| 	if client != nil && client.Core().GetRESTClient().GetRateLimiter() != nil {
 | |
| 		metrics.RegisterMetricAndTrackRateLimiterUsage("endpoint_controller", client.Core().GetRESTClient().GetRateLimiter())
 | |
| 	}
 | |
| 	e := &EndpointController{
 | |
| 		client: client,
 | |
| 		queue:  workqueue.New(),
 | |
| 	}
 | |
| 
 | |
| 	e.serviceStore.Store, e.serviceController = framework.NewInformer(
 | |
| 		&cache.ListWatch{
 | |
| 			ListFunc: func(options api.ListOptions) (runtime.Object, error) {
 | |
| 				return e.client.Core().Services(api.NamespaceAll).List(options)
 | |
| 			},
 | |
| 			WatchFunc: func(options api.ListOptions) (watch.Interface, error) {
 | |
| 				return e.client.Core().Services(api.NamespaceAll).Watch(options)
 | |
| 			},
 | |
| 		},
 | |
| 		&api.Service{},
 | |
| 		// TODO: Can we have much longer period here?
 | |
| 		FullServiceResyncPeriod,
 | |
| 		framework.ResourceEventHandlerFuncs{
 | |
| 			AddFunc: e.enqueueService,
 | |
| 			UpdateFunc: func(old, cur interface{}) {
 | |
| 				e.enqueueService(cur)
 | |
| 			},
 | |
| 			DeleteFunc: e.enqueueService,
 | |
| 		},
 | |
| 	)
 | |
| 
 | |
| 	podInformer.AddEventHandler(framework.ResourceEventHandlerFuncs{
 | |
| 		AddFunc:    e.addPod,
 | |
| 		UpdateFunc: e.updatePod,
 | |
| 		DeleteFunc: e.deletePod,
 | |
| 	})
 | |
| 	e.podStore.Indexer = podInformer.GetIndexer()
 | |
| 	e.podController = podInformer.GetController()
 | |
| 	e.podStoreSynced = podInformer.HasSynced
 | |
| 
 | |
| 	return e
 | |
| }
 | |
| 
 | |
| // NewEndpointControllerFromClient returns a new *EndpointController that runs its own informer.
 | |
| func NewEndpointControllerFromClient(client *clientset.Clientset, resyncPeriod controller.ResyncPeriodFunc) *EndpointController {
 | |
| 	podInformer := informers.NewPodInformer(client, resyncPeriod())
 | |
| 	e := NewEndpointController(podInformer, client)
 | |
| 	e.internalPodInformer = podInformer
 | |
| 
 | |
| 	return e
 | |
| }
 | |
| 
 | |
| // EndpointController manages selector-based service endpoints.
 | |
| type EndpointController struct {
 | |
| 	client *clientset.Clientset
 | |
| 
 | |
| 	serviceStore cache.StoreToServiceLister
 | |
| 	podStore     cache.StoreToPodLister
 | |
| 
 | |
| 	// internalPodInformer is used to hold a personal informer.  If we're using
 | |
| 	// a normal shared informer, then the informer will be started for us.  If
 | |
| 	// we have a personal informer, we must start it ourselves.   If you start
 | |
| 	// the controller using NewEndpointController(passing SharedInformer), this
 | |
| 	// will be null
 | |
| 	internalPodInformer framework.SharedIndexInformer
 | |
| 
 | |
| 	// Services that need to be updated. A channel is inappropriate here,
 | |
| 	// because it allows services with lots of pods to be serviced much
 | |
| 	// more often than services with few pods; it also would cause a
 | |
| 	// service that's inserted multiple times to be processed more than
 | |
| 	// necessary.
 | |
| 	queue *workqueue.Type
 | |
| 
 | |
| 	// Since we join two objects, we'll watch both of them with
 | |
| 	// controllers.
 | |
| 	serviceController *framework.Controller
 | |
| 	podController     framework.ControllerInterface
 | |
| 	// podStoreSynced returns true if the pod store has been synced at least once.
 | |
| 	// Added as a member to the struct to allow injection for testing.
 | |
| 	podStoreSynced func() bool
 | |
| }
 | |
| 
 | |
| // Runs e; will not return until stopCh is closed. workers determines how many
 | |
| // endpoints will be handled in parallel.
 | |
| func (e *EndpointController) Run(workers int, stopCh <-chan struct{}) {
 | |
| 	defer utilruntime.HandleCrash()
 | |
| 	go e.serviceController.Run(stopCh)
 | |
| 	go e.podController.Run(stopCh)
 | |
| 	for i := 0; i < workers; i++ {
 | |
| 		go wait.Until(e.worker, time.Second, stopCh)
 | |
| 	}
 | |
| 	go func() {
 | |
| 		defer utilruntime.HandleCrash()
 | |
| 		time.Sleep(5 * time.Minute) // give time for our cache to fill
 | |
| 		e.checkLeftoverEndpoints()
 | |
| 	}()
 | |
| 
 | |
| 	if e.internalPodInformer != nil {
 | |
| 		go e.internalPodInformer.Run(stopCh)
 | |
| 	}
 | |
| 
 | |
| 	<-stopCh
 | |
| 	e.queue.ShutDown()
 | |
| }
 | |
| 
 | |
| func (e *EndpointController) getPodServiceMemberships(pod *api.Pod) (sets.String, error) {
 | |
| 	set := sets.String{}
 | |
| 	services, err := e.serviceStore.GetPodServices(pod)
 | |
| 	if err != nil {
 | |
| 		// don't log this error because this function makes pointless
 | |
| 		// errors when no services match.
 | |
| 		return set, nil
 | |
| 	}
 | |
| 	for i := range services {
 | |
| 		key, err := keyFunc(&services[i])
 | |
| 		if err != nil {
 | |
| 			return nil, err
 | |
| 		}
 | |
| 		set.Insert(key)
 | |
| 	}
 | |
| 	return set, nil
 | |
| }
 | |
| 
 | |
| // When a pod is added, figure out what services it will be a member of and
 | |
| // enqueue them. obj must have *api.Pod type.
 | |
| func (e *EndpointController) addPod(obj interface{}) {
 | |
| 	pod := obj.(*api.Pod)
 | |
| 	services, err := e.getPodServiceMemberships(pod)
 | |
| 	if err != nil {
 | |
| 		glog.Errorf("Unable to get pod %v/%v's service memberships: %v", pod.Namespace, pod.Name, err)
 | |
| 		return
 | |
| 	}
 | |
| 	for key := range services {
 | |
| 		e.queue.Add(key)
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // When a pod is updated, figure out what services it used to be a member of
 | |
| // and what services it will be a member of, and enqueue the union of these.
 | |
| // old and cur must be *api.Pod types.
 | |
| func (e *EndpointController) updatePod(old, cur interface{}) {
 | |
| 	newPod := cur.(*api.Pod)
 | |
| 	oldPod := old.(*api.Pod)
 | |
| 	if newPod.ResourceVersion == oldPod.ResourceVersion {
 | |
| 		// Periodic resync will send update events for all known pods.
 | |
| 		// Two different versions of the same pod will always have different RVs.
 | |
| 		return
 | |
| 	}
 | |
| 	services, err := e.getPodServiceMemberships(newPod)
 | |
| 	if err != nil {
 | |
| 		glog.Errorf("Unable to get pod %v/%v's service memberships: %v", newPod.Namespace, newPod.Name, err)
 | |
| 		return
 | |
| 	}
 | |
| 
 | |
| 	// Only need to get the old services if the labels changed.
 | |
| 	if !reflect.DeepEqual(newPod.Labels, oldPod.Labels) ||
 | |
| 		!hostNameAndDomainAreEqual(newPod, oldPod) {
 | |
| 		oldServices, err := e.getPodServiceMemberships(oldPod)
 | |
| 		if err != nil {
 | |
| 			glog.Errorf("Unable to get pod %v/%v's service memberships: %v", oldPod.Namespace, oldPod.Name, err)
 | |
| 			return
 | |
| 		}
 | |
| 		services = services.Union(oldServices)
 | |
| 	}
 | |
| 	for key := range services {
 | |
| 		e.queue.Add(key)
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func hostNameAndDomainAreEqual(pod1, pod2 *api.Pod) bool {
 | |
| 	return getHostname(pod1) == getHostname(pod2) &&
 | |
| 		getSubdomain(pod1) == getSubdomain(pod2)
 | |
| }
 | |
| 
 | |
| func getHostname(pod *api.Pod) string {
 | |
| 	if len(pod.Spec.Hostname) > 0 {
 | |
| 		return pod.Spec.Hostname
 | |
| 	}
 | |
| 	if pod.Annotations != nil {
 | |
| 		return pod.Annotations[utilpod.PodHostnameAnnotation]
 | |
| 	}
 | |
| 	return ""
 | |
| }
 | |
| 
 | |
| func getSubdomain(pod *api.Pod) string {
 | |
| 	if len(pod.Spec.Subdomain) > 0 {
 | |
| 		return pod.Spec.Subdomain
 | |
| 	}
 | |
| 	if pod.Annotations != nil {
 | |
| 		return pod.Annotations[utilpod.PodSubdomainAnnotation]
 | |
| 	}
 | |
| 	return ""
 | |
| }
 | |
| 
 | |
| // When a pod is deleted, enqueue the services the pod used to be a member of.
 | |
| // obj could be an *api.Pod, or a DeletionFinalStateUnknown marker item.
 | |
| func (e *EndpointController) deletePod(obj interface{}) {
 | |
| 	if _, ok := obj.(*api.Pod); ok {
 | |
| 		// Enqueue all the services that the pod used to be a member
 | |
| 		// of. This happens to be exactly the same thing we do when a
 | |
| 		// pod is added.
 | |
| 		e.addPod(obj)
 | |
| 		return
 | |
| 	}
 | |
| 	podKey, err := keyFunc(obj)
 | |
| 	if err != nil {
 | |
| 		glog.Errorf("Couldn't get key for object %#v: %v", obj, err)
 | |
| 		return
 | |
| 	}
 | |
| 	glog.Infof("Pod %q was deleted but we don't have a record of its final state, so it will take up to %v before it will be removed from all endpoint records.", podKey, FullServiceResyncPeriod)
 | |
| 
 | |
| 	// TODO: keep a map of pods to services to handle this condition.
 | |
| }
 | |
| 
 | |
| // obj could be an *api.Service, or a DeletionFinalStateUnknown marker item.
 | |
| func (e *EndpointController) enqueueService(obj interface{}) {
 | |
| 	key, err := keyFunc(obj)
 | |
| 	if err != nil {
 | |
| 		glog.Errorf("Couldn't get key for object %+v: %v", obj, err)
 | |
| 		return
 | |
| 	}
 | |
| 
 | |
| 	e.queue.Add(key)
 | |
| }
 | |
| 
 | |
| // worker runs a worker thread that just dequeues items, processes them, and
 | |
| // marks them done. You may run as many of these in parallel as you wish; the
 | |
| // workqueue guarantees that they will not end up processing the same service
 | |
| // at the same time.
 | |
| func (e *EndpointController) worker() {
 | |
| 	for {
 | |
| 		func() {
 | |
| 			key, quit := e.queue.Get()
 | |
| 			if quit {
 | |
| 				return
 | |
| 			}
 | |
| 			// Use defer: in the unlikely event that there's a
 | |
| 			// panic, we'd still like this to get marked done--
 | |
| 			// otherwise the controller will not be able to sync
 | |
| 			// this service again until it is restarted.
 | |
| 			defer e.queue.Done(key)
 | |
| 			e.syncService(key.(string))
 | |
| 		}()
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func (e *EndpointController) syncService(key string) {
 | |
| 	startTime := time.Now()
 | |
| 	defer func() {
 | |
| 		glog.V(4).Infof("Finished syncing service %q endpoints. (%v)", key, time.Now().Sub(startTime))
 | |
| 	}()
 | |
| 
 | |
| 	if !e.podStoreSynced() {
 | |
| 		// Sleep so we give the pod reflector goroutine a chance to run.
 | |
| 		time.Sleep(PodStoreSyncedPollPeriod)
 | |
| 		glog.Infof("Waiting for pods controller to sync, requeuing service %v", key)
 | |
| 		e.queue.Add(key)
 | |
| 		return
 | |
| 	}
 | |
| 
 | |
| 	obj, exists, err := e.serviceStore.Store.GetByKey(key)
 | |
| 	if err != nil || !exists {
 | |
| 		// Delete the corresponding endpoint, as the service has been deleted.
 | |
| 		// TODO: Please note that this will delete an endpoint when a
 | |
| 		// service is deleted. However, if we're down at the time when
 | |
| 		// the service is deleted, we will miss that deletion, so this
 | |
| 		// doesn't completely solve the problem. See #6877.
 | |
| 		namespace, name, err := cache.SplitMetaNamespaceKey(key)
 | |
| 		if err != nil {
 | |
| 			glog.Errorf("Need to delete endpoint with key %q, but couldn't understand the key: %v", key, err)
 | |
| 			// Don't retry, as the key isn't going to magically become understandable.
 | |
| 			return
 | |
| 		}
 | |
| 		err = e.client.Endpoints(namespace).Delete(name, nil)
 | |
| 		if err != nil && !errors.IsNotFound(err) {
 | |
| 			glog.Errorf("Error deleting endpoint %q: %v", key, err)
 | |
| 			e.queue.Add(key) // Retry
 | |
| 		}
 | |
| 		return
 | |
| 	}
 | |
| 
 | |
| 	service := obj.(*api.Service)
 | |
| 	if service.Spec.Selector == nil {
 | |
| 		// services without a selector receive no endpoints from this controller;
 | |
| 		// these services will receive the endpoints that are created out-of-band via the REST API.
 | |
| 		return
 | |
| 	}
 | |
| 
 | |
| 	glog.V(5).Infof("About to update endpoints for service %q", key)
 | |
| 	pods, err := e.podStore.Pods(service.Namespace).List(labels.Set(service.Spec.Selector).AsSelector())
 | |
| 	if err != nil {
 | |
| 		// Since we're getting stuff from a local cache, it is
 | |
| 		// basically impossible to get this error.
 | |
| 		glog.Errorf("Error syncing service %q: %v", key, err)
 | |
| 		e.queue.Add(key) // Retry
 | |
| 		return
 | |
| 	}
 | |
| 
 | |
| 	subsets := []api.EndpointSubset{}
 | |
| 	podHostNames := map[string]endpoints.HostRecord{}
 | |
| 
 | |
| 	var tolerateUnreadyEndpoints bool
 | |
| 	if v, ok := service.Annotations[TolerateUnreadyEndpointsAnnotation]; ok {
 | |
| 		b, err := strconv.ParseBool(v)
 | |
| 		if err == nil {
 | |
| 			tolerateUnreadyEndpoints = b
 | |
| 		} else {
 | |
| 			glog.Errorf("Failed to parse annotation %v: %v", TolerateUnreadyEndpointsAnnotation, err)
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	for i := range pods.Items {
 | |
| 		pod := &pods.Items[i]
 | |
| 
 | |
| 		for i := range service.Spec.Ports {
 | |
| 			servicePort := &service.Spec.Ports[i]
 | |
| 
 | |
| 			portName := servicePort.Name
 | |
| 			portProto := servicePort.Protocol
 | |
| 			portNum, err := podutil.FindPort(pod, servicePort)
 | |
| 			if err != nil {
 | |
| 				glog.V(4).Infof("Failed to find port for service %s/%s: %v", service.Namespace, service.Name, err)
 | |
| 				continue
 | |
| 			}
 | |
| 			if len(pod.Status.PodIP) == 0 {
 | |
| 				glog.V(5).Infof("Failed to find an IP for pod %s/%s", pod.Namespace, pod.Name)
 | |
| 				continue
 | |
| 			}
 | |
| 			if pod.DeletionTimestamp != nil {
 | |
| 				glog.V(5).Infof("Pod is being deleted %s/%s", pod.Namespace, pod.Name)
 | |
| 				continue
 | |
| 			}
 | |
| 
 | |
| 			epp := api.EndpointPort{Name: portName, Port: int32(portNum), Protocol: portProto}
 | |
| 			epa := api.EndpointAddress{
 | |
| 				IP:       pod.Status.PodIP,
 | |
| 				NodeName: &pod.Spec.NodeName,
 | |
| 				TargetRef: &api.ObjectReference{
 | |
| 					Kind:            "Pod",
 | |
| 					Namespace:       pod.ObjectMeta.Namespace,
 | |
| 					Name:            pod.ObjectMeta.Name,
 | |
| 					UID:             pod.ObjectMeta.UID,
 | |
| 					ResourceVersion: pod.ObjectMeta.ResourceVersion,
 | |
| 				}}
 | |
| 
 | |
| 			hostname := getHostname(pod)
 | |
| 			if len(hostname) > 0 &&
 | |
| 				getSubdomain(pod) == service.Name &&
 | |
| 				service.Namespace == pod.Namespace {
 | |
| 				hostRecord := endpoints.HostRecord{
 | |
| 					HostName: hostname,
 | |
| 				}
 | |
| 				// TODO: stop populating podHostNames annotation in 1.4
 | |
| 				podHostNames[string(pod.Status.PodIP)] = hostRecord
 | |
| 				epa.Hostname = hostname
 | |
| 			}
 | |
| 
 | |
| 			if tolerateUnreadyEndpoints || api.IsPodReady(pod) {
 | |
| 				subsets = append(subsets, api.EndpointSubset{
 | |
| 					Addresses: []api.EndpointAddress{epa},
 | |
| 					Ports:     []api.EndpointPort{epp},
 | |
| 				})
 | |
| 			} else {
 | |
| 				glog.V(5).Infof("Pod is out of service: %v/%v", pod.Namespace, pod.Name)
 | |
| 				subsets = append(subsets, api.EndpointSubset{
 | |
| 					NotReadyAddresses: []api.EndpointAddress{epa},
 | |
| 					Ports:             []api.EndpointPort{epp},
 | |
| 				})
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 	subsets = endpoints.RepackSubsets(subsets)
 | |
| 
 | |
| 	// See if there's actually an update here.
 | |
| 	currentEndpoints, err := e.client.Endpoints(service.Namespace).Get(service.Name)
 | |
| 	if err != nil {
 | |
| 		if errors.IsNotFound(err) {
 | |
| 			currentEndpoints = &api.Endpoints{
 | |
| 				ObjectMeta: api.ObjectMeta{
 | |
| 					Name:   service.Name,
 | |
| 					Labels: service.Labels,
 | |
| 				},
 | |
| 			}
 | |
| 		} else {
 | |
| 			glog.Errorf("Error getting endpoints: %v", err)
 | |
| 			e.queue.Add(key) // Retry
 | |
| 			return
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	serializedPodHostNames := ""
 | |
| 	if len(podHostNames) > 0 {
 | |
| 		b, err := json.Marshal(podHostNames)
 | |
| 		if err != nil {
 | |
| 			glog.Errorf("Error updating endpoints. Marshalling of hostnames failed.: %v", err)
 | |
| 			e.queue.Add(key) // Retry
 | |
| 			return
 | |
| 		}
 | |
| 		serializedPodHostNames = string(b)
 | |
| 	}
 | |
| 
 | |
| 	newAnnotations := make(map[string]string)
 | |
| 	newAnnotations[endpoints.PodHostnamesAnnotation] = serializedPodHostNames
 | |
| 	if reflect.DeepEqual(currentEndpoints.Subsets, subsets) &&
 | |
| 		reflect.DeepEqual(currentEndpoints.Labels, service.Labels) {
 | |
| 		glog.V(5).Infof("endpoints are equal for %s/%s, skipping update", service.Namespace, service.Name)
 | |
| 		return
 | |
| 	}
 | |
| 	newEndpoints := currentEndpoints
 | |
| 	newEndpoints.Subsets = subsets
 | |
| 	newEndpoints.Labels = service.Labels
 | |
| 	if newEndpoints.Annotations == nil {
 | |
| 		newEndpoints.Annotations = make(map[string]string)
 | |
| 	}
 | |
| 	if len(serializedPodHostNames) == 0 {
 | |
| 		delete(newEndpoints.Annotations, endpoints.PodHostnamesAnnotation)
 | |
| 	} else {
 | |
| 		newEndpoints.Annotations[endpoints.PodHostnamesAnnotation] = serializedPodHostNames
 | |
| 	}
 | |
| 
 | |
| 	createEndpoints := len(currentEndpoints.ResourceVersion) == 0
 | |
| 	if createEndpoints {
 | |
| 		// No previous endpoints, create them
 | |
| 		_, err = e.client.Endpoints(service.Namespace).Create(newEndpoints)
 | |
| 	} else {
 | |
| 		// Pre-existing
 | |
| 		_, err = e.client.Endpoints(service.Namespace).Update(newEndpoints)
 | |
| 	}
 | |
| 	if err != nil {
 | |
| 		if createEndpoints && errors.IsForbidden(err) {
 | |
| 			// A request is forbidden primarily for two reasons:
 | |
| 			// 1. namespace is terminating, endpoint creation is not allowed by default.
 | |
| 			// 2. policy is misconfigured, in which case no service would function anywhere.
 | |
| 			// Given the frequency of 1, we log at a lower level.
 | |
| 			glog.V(5).Infof("Forbidden from creating endpoints: %v", err)
 | |
| 		} else {
 | |
| 			utilruntime.HandleError(err)
 | |
| 		}
 | |
| 		e.queue.Add(key) // Retry
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // checkLeftoverEndpoints lists all currently existing endpoints and adds their
 | |
| // service to the queue. This will detect endpoints that exist with no
 | |
| // corresponding service; these endpoints need to be deleted. We only need to
 | |
| // do this once on startup, because in steady-state these are detected (but
 | |
| // some stragglers could have been left behind if the endpoint controller
 | |
| // reboots).
 | |
| func (e *EndpointController) checkLeftoverEndpoints() {
 | |
| 	list, err := e.client.Endpoints(api.NamespaceAll).List(api.ListOptions{})
 | |
| 	if err != nil {
 | |
| 		glog.Errorf("Unable to list endpoints (%v); orphaned endpoints will not be cleaned up. (They're pretty harmless, but you can restart this component if you want another attempt made.)", err)
 | |
| 		return
 | |
| 	}
 | |
| 	for i := range list.Items {
 | |
| 		ep := &list.Items[i]
 | |
| 		key, err := keyFunc(ep)
 | |
| 		if err != nil {
 | |
| 			glog.Errorf("Unable to get key for endpoint %#v", ep)
 | |
| 			continue
 | |
| 		}
 | |
| 		e.queue.Add(key)
 | |
| 	}
 | |
| }
 |