2014-06-06 23:40:48 +00:00
|
|
|
/*
|
2016-06-03 00:25:58 +00:00
|
|
|
Copyright 2014 The Kubernetes Authors.
|
2014-06-06 23:40:48 +00:00
|
|
|
|
|
|
|
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 config
|
|
|
|
|
|
|
|
import (
|
2017-03-17 12:53:46 +00:00
|
|
|
"fmt"
|
|
|
|
"time"
|
2014-06-06 23:40:48 +00:00
|
|
|
|
2015-08-05 22:05:17 +00:00
|
|
|
"github.com/golang/glog"
|
2017-03-17 12:53:46 +00:00
|
|
|
"k8s.io/apimachinery/pkg/labels"
|
|
|
|
utilruntime "k8s.io/apimachinery/pkg/util/runtime"
|
|
|
|
"k8s.io/client-go/tools/cache"
|
2015-08-05 22:03:47 +00:00
|
|
|
"k8s.io/kubernetes/pkg/api"
|
2017-04-03 18:34:29 +00:00
|
|
|
coreinformers "k8s.io/kubernetes/pkg/client/informers/informers_generated/internalversion/core/internalversion"
|
2017-03-17 12:53:46 +00:00
|
|
|
listers "k8s.io/kubernetes/pkg/client/listers/core/internalversion"
|
2017-04-12 19:49:17 +00:00
|
|
|
"k8s.io/kubernetes/pkg/controller"
|
2015-08-05 22:03:47 +00:00
|
|
|
"k8s.io/kubernetes/pkg/util/config"
|
2014-06-06 23:40:48 +00:00
|
|
|
)
|
|
|
|
|
2014-07-15 13:03:08 +00:00
|
|
|
// ServiceConfigHandler is an abstract interface of objects which receive update notifications for the set of services.
|
2014-06-06 23:40:48 +00:00
|
|
|
type ServiceConfigHandler interface {
|
2017-04-01 04:48:39 +00:00
|
|
|
// OnServiceUpdate gets called when a service is created, removed or changed
|
|
|
|
// on any of the configuration sources. An example is when a new service
|
|
|
|
// comes up.
|
|
|
|
//
|
|
|
|
// NOTE: For efficiency, services are being passed by reference, thus,
|
|
|
|
// OnServiceUpdate should NOT modify pointers of a given slice.
|
|
|
|
// Those service objects are shared with other layers of the system and
|
|
|
|
// are guaranteed to be immutable with the assumption that are also
|
|
|
|
// not mutated by those handlers. Make a deep copy if you need to modify
|
|
|
|
// them in your code.
|
|
|
|
OnServiceUpdate(services []*api.Service)
|
2014-06-06 23:40:48 +00:00
|
|
|
}
|
|
|
|
|
2017-04-04 11:00:58 +00:00
|
|
|
// EndpointsHandler is an abstract interface o objects which receive
|
|
|
|
// notifications about endpoints object changes.
|
|
|
|
type EndpointsHandler interface {
|
|
|
|
// OnEndpointsAdd is called whenever creation of new endpoints object
|
|
|
|
// is observed.
|
|
|
|
OnEndpointsAdd(endpoints *api.Endpoints)
|
|
|
|
// OnEndpointsUpdate is called whenever modification of an existing
|
|
|
|
// endpoints object is observed.
|
|
|
|
OnEndpointsUpdate(oldEndpoints, endpoints *api.Endpoints)
|
|
|
|
// OnEndpointsDelete is called whever deletion of an existing endpoints
|
|
|
|
// object is observed.
|
|
|
|
OnEndpointsDelete(endpoints *api.Endpoints)
|
|
|
|
// OnEndpointsSynced is called once all the initial event handlers were
|
|
|
|
// called and the state is fully propagated to local cache.
|
|
|
|
OnEndpointsSynced()
|
|
|
|
}
|
|
|
|
|
2014-07-08 16:55:11 +00:00
|
|
|
// EndpointsConfig tracks a set of endpoints configurations.
|
|
|
|
// It accepts "set", "add" and "remove" operations of endpoints via channels, and invokes registered handlers on change.
|
|
|
|
type EndpointsConfig struct {
|
2017-04-04 11:00:58 +00:00
|
|
|
lister listers.EndpointsLister
|
|
|
|
listerSynced cache.InformerSynced
|
|
|
|
eventHandlers []EndpointsHandler
|
2014-07-08 16:55:11 +00:00
|
|
|
}
|
2014-06-06 23:40:48 +00:00
|
|
|
|
2014-09-02 10:00:28 +00:00
|
|
|
// NewEndpointsConfig creates a new EndpointsConfig.
|
2017-04-03 18:34:29 +00:00
|
|
|
func NewEndpointsConfig(endpointsInformer coreinformers.EndpointsInformer, resyncPeriod time.Duration) *EndpointsConfig {
|
|
|
|
result := &EndpointsConfig{
|
|
|
|
lister: endpointsInformer.Lister(),
|
|
|
|
listerSynced: endpointsInformer.Informer().HasSynced,
|
|
|
|
}
|
2017-03-17 12:53:46 +00:00
|
|
|
|
2017-04-03 18:34:29 +00:00
|
|
|
endpointsInformer.Informer().AddEventHandlerWithResyncPeriod(
|
2017-03-17 12:53:46 +00:00
|
|
|
cache.ResourceEventHandlerFuncs{
|
|
|
|
AddFunc: result.handleAddEndpoints,
|
|
|
|
UpdateFunc: result.handleUpdateEndpoints,
|
|
|
|
DeleteFunc: result.handleDeleteEndpoints,
|
|
|
|
},
|
2017-04-03 18:34:29 +00:00
|
|
|
resyncPeriod,
|
2017-03-17 12:53:46 +00:00
|
|
|
)
|
2017-04-03 18:34:29 +00:00
|
|
|
|
2017-03-17 12:53:46 +00:00
|
|
|
return result
|
2014-07-08 16:55:11 +00:00
|
|
|
}
|
2014-06-06 23:40:48 +00:00
|
|
|
|
2017-04-04 11:00:58 +00:00
|
|
|
// RegisterEventHandler registers a handler which is called on every endpoints change.
|
|
|
|
func (c *EndpointsConfig) RegisterEventHandler(handler EndpointsHandler) {
|
|
|
|
c.eventHandlers = append(c.eventHandlers, handler)
|
|
|
|
}
|
|
|
|
|
2017-04-03 18:34:29 +00:00
|
|
|
// Run starts the goroutine responsible for calling registered handlers.
|
2017-03-17 12:53:46 +00:00
|
|
|
func (c *EndpointsConfig) Run(stopCh <-chan struct{}) {
|
2017-04-12 19:49:17 +00:00
|
|
|
defer utilruntime.HandleCrash()
|
|
|
|
|
|
|
|
glog.Info("Starting endpoints config controller")
|
|
|
|
defer glog.Info("Shutting down endpoints config controller")
|
|
|
|
|
|
|
|
if !controller.WaitForCacheSync("endpoints config", stopCh, c.listerSynced) {
|
2017-03-17 12:53:46 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2017-04-11 07:59:31 +00:00
|
|
|
for i := range c.eventHandlers {
|
|
|
|
glog.V(3).Infof("Calling handler.OnEndpointsSynced()")
|
|
|
|
c.eventHandlers[i].OnEndpointsSynced()
|
|
|
|
}
|
|
|
|
|
|
|
|
<-stopCh
|
2014-07-08 16:55:11 +00:00
|
|
|
}
|
2014-06-06 23:40:48 +00:00
|
|
|
|
2017-04-04 11:00:58 +00:00
|
|
|
func (c *EndpointsConfig) handleAddEndpoints(obj interface{}) {
|
|
|
|
endpoints, ok := obj.(*api.Endpoints)
|
|
|
|
if !ok {
|
|
|
|
utilruntime.HandleError(fmt.Errorf("unexpected object type: %v", obj))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
for i := range c.eventHandlers {
|
|
|
|
glog.V(4).Infof("Calling handler.OnEndpointsAdd")
|
|
|
|
c.eventHandlers[i].OnEndpointsAdd(endpoints)
|
|
|
|
}
|
2014-06-06 23:40:48 +00:00
|
|
|
}
|
|
|
|
|
2017-04-04 11:00:58 +00:00
|
|
|
func (c *EndpointsConfig) handleUpdateEndpoints(oldObj, newObj interface{}) {
|
|
|
|
oldEndpoints, ok := oldObj.(*api.Endpoints)
|
|
|
|
if !ok {
|
|
|
|
utilruntime.HandleError(fmt.Errorf("unexpected object type: %v", oldObj))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
endpoints, ok := newObj.(*api.Endpoints)
|
|
|
|
if !ok {
|
|
|
|
utilruntime.HandleError(fmt.Errorf("unexpected object type: %v", newObj))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
for i := range c.eventHandlers {
|
|
|
|
glog.V(4).Infof("Calling handler.OnEndpointsUpdate")
|
|
|
|
c.eventHandlers[i].OnEndpointsUpdate(oldEndpoints, endpoints)
|
|
|
|
}
|
2014-06-06 23:40:48 +00:00
|
|
|
}
|
|
|
|
|
2017-04-04 11:00:58 +00:00
|
|
|
func (c *EndpointsConfig) handleDeleteEndpoints(obj interface{}) {
|
|
|
|
endpoints, ok := obj.(*api.Endpoints)
|
|
|
|
if !ok {
|
|
|
|
tombstone, ok := obj.(cache.DeletedFinalStateUnknown)
|
|
|
|
if !ok {
|
|
|
|
utilruntime.HandleError(fmt.Errorf("unexpected object type: %v", obj))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if endpoints, ok = tombstone.Obj.(*api.Endpoints); !ok {
|
|
|
|
utilruntime.HandleError(fmt.Errorf("unexpected object type: %v", obj))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for i := range c.eventHandlers {
|
|
|
|
glog.V(4).Infof("Calling handler.OnEndpointsUpdate")
|
|
|
|
c.eventHandlers[i].OnEndpointsDelete(endpoints)
|
|
|
|
}
|
2014-06-06 23:40:48 +00:00
|
|
|
}
|
|
|
|
|
2014-07-08 16:55:11 +00:00
|
|
|
// ServiceConfig tracks a set of service configurations.
|
|
|
|
// It accepts "set", "add" and "remove" operations of services via channels, and invokes registered handlers on change.
|
|
|
|
type ServiceConfig struct {
|
2017-04-03 18:34:29 +00:00
|
|
|
lister listers.ServiceLister
|
|
|
|
listerSynced cache.InformerSynced
|
|
|
|
handlers []ServiceConfigHandler
|
2017-03-17 14:15:51 +00:00
|
|
|
// updates channel is used to trigger registered handlers
|
|
|
|
updates chan struct{}
|
2017-04-03 18:34:29 +00:00
|
|
|
stop chan struct{}
|
2014-07-08 16:55:11 +00:00
|
|
|
}
|
2014-06-06 23:40:48 +00:00
|
|
|
|
2014-07-08 16:55:11 +00:00
|
|
|
// NewServiceConfig creates a new ServiceConfig.
|
2017-04-03 18:34:29 +00:00
|
|
|
func NewServiceConfig(serviceInformer coreinformers.ServiceInformer, resyncPeriod time.Duration) *ServiceConfig {
|
|
|
|
result := &ServiceConfig{
|
|
|
|
lister: serviceInformer.Lister(),
|
|
|
|
listerSynced: serviceInformer.Informer().HasSynced,
|
|
|
|
// The updates channel is used to send interrupts to the Services handler.
|
|
|
|
// It's buffered because we never want to block for as long as there is a
|
|
|
|
// pending interrupt, but don't want to drop them if the handler is doing
|
|
|
|
// work.
|
|
|
|
updates: make(chan struct{}, 1),
|
|
|
|
stop: make(chan struct{}),
|
|
|
|
}
|
2017-03-17 14:15:51 +00:00
|
|
|
|
2017-04-03 18:34:29 +00:00
|
|
|
serviceInformer.Informer().AddEventHandlerWithResyncPeriod(
|
2017-03-17 14:15:51 +00:00
|
|
|
cache.ResourceEventHandlerFuncs{
|
|
|
|
AddFunc: result.handleAddService,
|
|
|
|
UpdateFunc: result.handleUpdateService,
|
|
|
|
DeleteFunc: result.handleDeleteService,
|
|
|
|
},
|
2017-04-03 18:34:29 +00:00
|
|
|
resyncPeriod,
|
2017-03-17 14:15:51 +00:00
|
|
|
)
|
2017-04-03 18:34:29 +00:00
|
|
|
|
2017-03-17 14:15:51 +00:00
|
|
|
return result
|
2014-07-08 16:55:11 +00:00
|
|
|
}
|
2014-06-06 23:40:48 +00:00
|
|
|
|
2017-02-27 16:38:59 +00:00
|
|
|
// RegisterHandler registers a handler which is called on every services change.
|
2014-07-08 16:55:11 +00:00
|
|
|
func (c *ServiceConfig) RegisterHandler(handler ServiceConfigHandler) {
|
2017-03-17 14:15:51 +00:00
|
|
|
c.handlers = append(c.handlers, handler)
|
2014-06-06 23:40:48 +00:00
|
|
|
}
|
|
|
|
|
2017-04-03 18:34:29 +00:00
|
|
|
// Run starts the goroutine responsible for calling
|
2017-03-17 14:29:44 +00:00
|
|
|
// registered handlers.
|
2017-03-17 14:15:51 +00:00
|
|
|
func (c *ServiceConfig) Run(stopCh <-chan struct{}) {
|
2017-04-12 19:49:17 +00:00
|
|
|
defer utilruntime.HandleCrash()
|
|
|
|
|
|
|
|
glog.Info("Starting service config controller")
|
|
|
|
defer glog.Info("Shutting down service config controller")
|
|
|
|
|
|
|
|
if !controller.WaitForCacheSync("service config", stopCh, c.listerSynced) {
|
2017-03-17 14:15:51 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2017-04-01 04:48:39 +00:00
|
|
|
// We have synced informers. Now we can start delivering updates
|
2017-03-17 14:15:51 +00:00
|
|
|
// to the registered handler.
|
2014-07-08 16:55:11 +00:00
|
|
|
go func() {
|
2017-04-12 19:49:17 +00:00
|
|
|
defer utilruntime.HandleCrash()
|
2017-04-03 18:34:29 +00:00
|
|
|
for {
|
|
|
|
select {
|
|
|
|
case <-c.updates:
|
|
|
|
services, err := c.lister.List(labels.Everything())
|
|
|
|
if err != nil {
|
|
|
|
glog.Errorf("Error while listing services from cache: %v", err)
|
|
|
|
// This will cause a retry (if there isnt' any other trigger in-flight).
|
|
|
|
c.dispatchUpdate()
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if services == nil {
|
|
|
|
services = []*api.Service{}
|
|
|
|
}
|
|
|
|
for i := range c.handlers {
|
|
|
|
glog.V(3).Infof("Calling handler.OnServiceUpdate()")
|
|
|
|
c.handlers[i].OnServiceUpdate(services)
|
|
|
|
}
|
|
|
|
case <-c.stop:
|
|
|
|
return
|
2017-03-17 14:15:51 +00:00
|
|
|
}
|
2014-07-08 16:55:11 +00:00
|
|
|
}
|
|
|
|
}()
|
2017-03-17 14:15:51 +00:00
|
|
|
// Close updates channel when stopCh is closed.
|
|
|
|
go func() {
|
|
|
|
<-stopCh
|
2017-04-03 18:34:29 +00:00
|
|
|
close(c.stop)
|
2017-03-17 14:15:51 +00:00
|
|
|
}()
|
2017-04-12 19:49:17 +00:00
|
|
|
|
|
|
|
<-stopCh
|
2014-06-06 23:40:48 +00:00
|
|
|
}
|
|
|
|
|
2017-03-17 14:15:51 +00:00
|
|
|
func (c *ServiceConfig) handleAddService(_ interface{}) {
|
|
|
|
c.dispatchUpdate()
|
2014-06-06 23:40:48 +00:00
|
|
|
}
|
|
|
|
|
2017-03-17 14:15:51 +00:00
|
|
|
func (c *ServiceConfig) handleUpdateService(_, _ interface{}) {
|
|
|
|
c.dispatchUpdate()
|
2014-06-06 23:40:48 +00:00
|
|
|
}
|
|
|
|
|
2017-03-17 14:15:51 +00:00
|
|
|
func (c *ServiceConfig) handleDeleteService(_ interface{}) {
|
|
|
|
c.dispatchUpdate()
|
2014-06-06 23:40:48 +00:00
|
|
|
}
|
|
|
|
|
2017-03-17 14:15:51 +00:00
|
|
|
func (c *ServiceConfig) dispatchUpdate() {
|
|
|
|
select {
|
|
|
|
case c.updates <- struct{}{}:
|
2017-04-03 18:34:29 +00:00
|
|
|
// Work enqueued successfully
|
|
|
|
case <-c.stop:
|
|
|
|
// We're shut down / avoid logging the message below
|
2017-03-17 14:15:51 +00:00
|
|
|
default:
|
2017-04-03 18:34:29 +00:00
|
|
|
glog.V(4).Infof("Service handler already has a pending interrupt.")
|
2014-06-06 23:40:48 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-12-04 07:54:40 +00:00
|
|
|
// watchForUpdates invokes bcaster.Notify() with the latest version of an object
|
2014-07-08 16:55:11 +00:00
|
|
|
// when changes occur.
|
2014-12-04 07:54:40 +00:00
|
|
|
func watchForUpdates(bcaster *config.Broadcaster, accessor config.Accessor, updates <-chan struct{}) {
|
2015-02-20 17:30:33 +00:00
|
|
|
for true {
|
|
|
|
<-updates
|
2014-12-04 07:54:40 +00:00
|
|
|
bcaster.Notify(accessor.MergedState())
|
2014-06-06 23:40:48 +00:00
|
|
|
}
|
|
|
|
}
|