mirror of https://github.com/k3s-io/k3s
123 lines
2.9 KiB
Go
123 lines
2.9 KiB
Go
|
package controller
|
||
|
|
||
|
import (
|
||
|
"context"
|
||
|
"sync"
|
||
|
"time"
|
||
|
|
||
|
"github.com/rancher/lasso/pkg/cache"
|
||
|
"github.com/rancher/lasso/pkg/client"
|
||
|
"k8s.io/apimachinery/pkg/runtime"
|
||
|
"k8s.io/apimachinery/pkg/runtime/schema"
|
||
|
cachetools "k8s.io/client-go/tools/cache"
|
||
|
)
|
||
|
|
||
|
type SharedControllerHandler interface {
|
||
|
OnChange(key string, obj runtime.Object) (runtime.Object, error)
|
||
|
}
|
||
|
|
||
|
type SharedController interface {
|
||
|
Controller
|
||
|
|
||
|
RegisterHandler(ctx context.Context, name string, handler SharedControllerHandler)
|
||
|
Client() *client.Client
|
||
|
}
|
||
|
|
||
|
type SharedControllerHandlerFunc func(key string, obj runtime.Object) (runtime.Object, error)
|
||
|
|
||
|
func (s SharedControllerHandlerFunc) OnChange(key string, obj runtime.Object) (runtime.Object, error) {
|
||
|
return s(key, obj)
|
||
|
}
|
||
|
|
||
|
type sharedController struct {
|
||
|
// this allows one to create a sharedcontroller but it will not actually be started
|
||
|
// unless some aspect of the controllers informer is accessed or needed to be used
|
||
|
deferredController func() (Controller, error)
|
||
|
sharedCacheFactory cache.SharedCacheFactory
|
||
|
controller Controller
|
||
|
gvk schema.GroupVersionKind
|
||
|
handler *SharedHandler
|
||
|
startLock sync.Mutex
|
||
|
started bool
|
||
|
startError error
|
||
|
client *client.Client
|
||
|
}
|
||
|
|
||
|
func (s *sharedController) Enqueue(namespace, name string) {
|
||
|
s.initController().Enqueue(namespace, name)
|
||
|
}
|
||
|
|
||
|
func (s *sharedController) EnqueueAfter(namespace, name string, delay time.Duration) {
|
||
|
s.initController().EnqueueAfter(namespace, name, delay)
|
||
|
}
|
||
|
|
||
|
func (s *sharedController) EnqueueKey(key string) {
|
||
|
s.initController().EnqueueKey(key)
|
||
|
}
|
||
|
|
||
|
func (s *sharedController) Informer() cachetools.SharedIndexInformer {
|
||
|
return s.initController().Informer()
|
||
|
}
|
||
|
|
||
|
func (s *sharedController) Client() *client.Client {
|
||
|
return s.client
|
||
|
}
|
||
|
|
||
|
func (s *sharedController) initController() Controller {
|
||
|
s.startLock.Lock()
|
||
|
defer s.startLock.Unlock()
|
||
|
|
||
|
if s.controller != nil {
|
||
|
return s.controller
|
||
|
}
|
||
|
|
||
|
controller, err := s.deferredController()
|
||
|
if err != nil {
|
||
|
controller = newErrorController()
|
||
|
}
|
||
|
|
||
|
s.startError = err
|
||
|
s.controller = controller
|
||
|
return s.controller
|
||
|
}
|
||
|
|
||
|
func (s *sharedController) Start(ctx context.Context, workers int) error {
|
||
|
s.startLock.Lock()
|
||
|
defer s.startLock.Unlock()
|
||
|
|
||
|
if s.startError != nil || s.controller == nil {
|
||
|
return s.startError
|
||
|
}
|
||
|
|
||
|
if err := s.controller.Start(ctx, workers); err != nil {
|
||
|
return err
|
||
|
}
|
||
|
s.started = true
|
||
|
|
||
|
go func() {
|
||
|
<-ctx.Done()
|
||
|
s.startLock.Lock()
|
||
|
defer s.startLock.Unlock()
|
||
|
s.started = false
|
||
|
}()
|
||
|
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func (s *sharedController) RegisterHandler(ctx context.Context, name string, handler SharedControllerHandler) {
|
||
|
// Ensure that controller is initialized
|
||
|
c := s.initController()
|
||
|
|
||
|
getHandlerTransaction(ctx).do(func() {
|
||
|
s.handler.Register(ctx, name, handler)
|
||
|
|
||
|
s.startLock.Lock()
|
||
|
defer s.startLock.Unlock()
|
||
|
if s.started {
|
||
|
for _, key := range c.Informer().GetStore().ListKeys() {
|
||
|
c.EnqueueKey(key)
|
||
|
}
|
||
|
}
|
||
|
})
|
||
|
}
|