k3s/vendor/github.com/rancher/lasso/pkg/controller/sharedcontroller.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)
}
}
})
}