mirror of
https://github.com/kyverno/kyverno.git
synced 2024-12-14 11:57:48 +00:00
2444b7c670
* refactor: add per resource reports aggregation Signed-off-by: Charles-Edouard Brétéché <charles.edouard@nirmata.com> * added controller implementation Signed-off-by: Charles-Edouard Brétéché <charles.edouard@nirmata.com> * clean Signed-off-by: Charles-Edouard Brétéché <charles.edouard@nirmata.com> * fix Signed-off-by: Charles-Edouard Brétéché <charles.edouard@nirmata.com> * fix Signed-off-by: Charles-Edouard Brétéché <charles.edouard@nirmata.com> * fix Signed-off-by: Charles-Edouard Brétéché <charles.edouard@nirmata.com> * fix kuttl tests Signed-off-by: Charles-Edouard Brétéché <charles.edouard@nirmata.com> * fix Signed-off-by: Charles-Edouard Brétéché <charles.edouard@nirmata.com> * fix kuttl tests Signed-off-by: Charles-Edouard Brétéché <charles.edouard@nirmata.com> * vaps Signed-off-by: Charles-Edouard Brétéché <charles.edouard@nirmata.com> --------- Signed-off-by: Charles-Edouard Brétéché <charles.edouard@nirmata.com>
215 lines
7.4 KiB
Go
215 lines
7.4 KiB
Go
package controller
|
|
|
|
import (
|
|
"errors"
|
|
"time"
|
|
|
|
"github.com/go-logr/logr"
|
|
kubeutils "github.com/kyverno/kyverno/pkg/utils/kube"
|
|
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
|
"k8s.io/client-go/tools/cache"
|
|
"k8s.io/client-go/util/workqueue"
|
|
)
|
|
|
|
type (
|
|
addFunc = addFuncT[interface{}]
|
|
updateFunc = updateFuncT[interface{}]
|
|
deleteFunc = deleteFuncT[interface{}]
|
|
addFuncT[T any] func(T)
|
|
updateFuncT[T any] func(T, T)
|
|
deleteFuncT[T any] func(T)
|
|
keyFunc = keyFuncT[interface{}, interface{}]
|
|
keyFuncT[T, U any] func(T) (U, error)
|
|
EnqueueFunc = EnqueueFuncT[interface{}]
|
|
EnqueueFuncT[T any] func(T) error
|
|
)
|
|
|
|
func AddEventHandlers(informer cache.SharedInformer, a addFunc, u updateFunc, d deleteFunc) (cache.ResourceEventHandlerRegistration, error) {
|
|
var onDelete deleteFunc
|
|
if d != nil {
|
|
onDelete = func(obj interface{}) {
|
|
d(kubeutils.GetObjectWithTombstone(obj))
|
|
}
|
|
}
|
|
return informer.AddEventHandler(cache.ResourceEventHandlerFuncs{
|
|
AddFunc: a,
|
|
UpdateFunc: u,
|
|
DeleteFunc: onDelete,
|
|
})
|
|
}
|
|
|
|
func AddEventHandlersT[T any](informer cache.SharedInformer, a addFuncT[T], u updateFuncT[T], d deleteFuncT[T]) (cache.ResourceEventHandlerRegistration, error) {
|
|
var onAdd addFunc
|
|
var onUpdate updateFunc
|
|
var onDelete deleteFunc
|
|
if a != nil {
|
|
onAdd = func(obj interface{}) { a(obj.(T)) }
|
|
}
|
|
if u != nil {
|
|
onUpdate = func(old, obj interface{}) { u(old.(T), obj.(T)) }
|
|
}
|
|
if d != nil {
|
|
onDelete = func(obj interface{}) { d(obj.(T)) }
|
|
}
|
|
return AddEventHandlers(informer, onAdd, onUpdate, onDelete)
|
|
}
|
|
|
|
func AddKeyedEventHandlers(logger logr.Logger, informer cache.SharedInformer, queue workqueue.RateLimitingInterface, parseKey keyFunc) (EnqueueFunc, cache.ResourceEventHandlerRegistration, error) {
|
|
enqueueFunc := LogError(logger, Parse(parseKey, Queue(queue)))
|
|
if registration, err := AddEventHandlers(informer, AddFunc(logger, enqueueFunc), UpdateFunc(logger, enqueueFunc), DeleteFunc(logger, enqueueFunc)); err != nil {
|
|
return nil, nil, err
|
|
} else {
|
|
return enqueueFunc, registration, nil
|
|
}
|
|
}
|
|
|
|
func AddKeyedEventHandlersT[K metav1.Object](logger logr.Logger, informer cache.SharedInformer, queue workqueue.RateLimitingInterface, parseKey keyFuncT[K, interface{}]) (EnqueueFuncT[K], cache.ResourceEventHandlerRegistration, error) {
|
|
enqueueFunc := LogError(logger, Parse(parseKey, Queue(queue)))
|
|
if registration, err := AddEventHandlersT(informer, AddFuncT(logger, enqueueFunc), UpdateFuncT(logger, enqueueFunc), DeleteFuncT(logger, enqueueFunc)); err != nil {
|
|
return nil, nil, err
|
|
} else {
|
|
return enqueueFunc, registration, nil
|
|
}
|
|
}
|
|
|
|
func AddDelayedKeyedEventHandlers(logger logr.Logger, informer cache.SharedInformer, queue workqueue.RateLimitingInterface, delay time.Duration, parseKey keyFunc) (EnqueueFunc, cache.ResourceEventHandlerRegistration, error) {
|
|
enqueueFunc := LogError(logger, Parse(parseKey, QueueAfter(queue, delay)))
|
|
if registration, err := AddEventHandlers(informer, AddFunc(logger, enqueueFunc), UpdateFunc(logger, enqueueFunc), DeleteFunc(logger, enqueueFunc)); err != nil {
|
|
return nil, nil, err
|
|
} else {
|
|
return enqueueFunc, registration, nil
|
|
}
|
|
}
|
|
|
|
func AddDefaultEventHandlers(logger logr.Logger, informer cache.SharedInformer, queue workqueue.RateLimitingInterface) (EnqueueFunc, cache.ResourceEventHandlerRegistration, error) {
|
|
return AddKeyedEventHandlers(logger, informer, queue, MetaNamespaceKey)
|
|
}
|
|
|
|
func AddDefaultEventHandlersT[K metav1.Object](logger logr.Logger, informer cache.SharedInformer, queue workqueue.RateLimitingInterface) (EnqueueFuncT[K], cache.ResourceEventHandlerRegistration, error) {
|
|
return AddKeyedEventHandlersT(logger, informer, queue, MetaNamespaceKeyT[K])
|
|
}
|
|
|
|
func AddDelayedDefaultEventHandlers(logger logr.Logger, informer cache.SharedInformer, queue workqueue.RateLimitingInterface, delay time.Duration) (EnqueueFunc, cache.ResourceEventHandlerRegistration, error) {
|
|
return AddDelayedKeyedEventHandlers(logger, informer, queue, delay, MetaNamespaceKey)
|
|
}
|
|
|
|
func AddExplicitEventHandlers[K any](logger logr.Logger, informer cache.SharedInformer, queue workqueue.RateLimitingInterface, parseKey func(K) cache.ExplicitKey) (EnqueueFunc, cache.ResourceEventHandlerRegistration, error) {
|
|
return AddKeyedEventHandlers(logger, informer, queue, ExplicitKey(parseKey))
|
|
}
|
|
|
|
func AddDelayedExplicitEventHandlers[K any](logger logr.Logger, informer cache.SharedInformer, queue workqueue.RateLimitingInterface, delay time.Duration, parseKey func(K) cache.ExplicitKey) (EnqueueFunc, cache.ResourceEventHandlerRegistration, error) {
|
|
return AddDelayedKeyedEventHandlers(logger, informer, queue, delay, ExplicitKey(parseKey))
|
|
}
|
|
|
|
func LogError[K any](logger logr.Logger, inner EnqueueFuncT[K]) EnqueueFuncT[K] {
|
|
return func(obj K) error {
|
|
err := inner(obj)
|
|
if err != nil {
|
|
logger.Error(err, "failed to compute key name", "obj", obj)
|
|
}
|
|
return err
|
|
}
|
|
}
|
|
|
|
func Parse[K, L any](parseKey keyFuncT[K, L], inner EnqueueFuncT[L]) EnqueueFuncT[K] {
|
|
return func(obj K) error {
|
|
if key, err := parseKey(obj); err != nil {
|
|
return err
|
|
} else {
|
|
return inner(key)
|
|
}
|
|
}
|
|
}
|
|
|
|
func Queue(queue workqueue.RateLimitingInterface) EnqueueFunc {
|
|
return func(obj interface{}) error {
|
|
queue.Add(obj)
|
|
return nil
|
|
}
|
|
}
|
|
|
|
func QueueAfter(queue workqueue.RateLimitingInterface, delay time.Duration) EnqueueFunc {
|
|
return func(obj interface{}) error {
|
|
queue.AddAfter(obj, delay)
|
|
return nil
|
|
}
|
|
}
|
|
|
|
func MetaObjectToName(obj metav1.Object) string {
|
|
return cache.MetaObjectToName(obj).String()
|
|
}
|
|
|
|
func MetaNamespaceKey(obj interface{}) (interface{}, error) {
|
|
return cache.MetaNamespaceKeyFunc(obj)
|
|
}
|
|
|
|
func MetaNamespaceKeyT[T any](obj T) (interface{}, error) {
|
|
return MetaNamespaceKey(obj)
|
|
}
|
|
|
|
func ExplicitKey[K any](parseKey func(K) cache.ExplicitKey) keyFunc {
|
|
return func(obj interface{}) (interface{}, error) {
|
|
if obj == nil {
|
|
return nil, errors.New("obj is nil")
|
|
}
|
|
if key, ok := obj.(K); !ok {
|
|
return nil, errors.New("obj cannot be converted")
|
|
} else {
|
|
return parseKey(key), nil
|
|
}
|
|
}
|
|
}
|
|
|
|
func AddFunc(logger logr.Logger, enqueue EnqueueFunc) addFunc {
|
|
return func(obj interface{}) {
|
|
if err := enqueue(obj); err != nil {
|
|
logger.Error(err, "failed to enqueue object", "obj", obj)
|
|
}
|
|
}
|
|
}
|
|
|
|
func UpdateFunc(logger logr.Logger, enqueue EnqueueFunc) updateFunc {
|
|
return func(old, obj interface{}) {
|
|
oldMeta := old.(metav1.Object)
|
|
objMeta := obj.(metav1.Object)
|
|
if oldMeta.GetResourceVersion() != objMeta.GetResourceVersion() {
|
|
if err := enqueue(obj); err != nil {
|
|
logger.Error(err, "failed to enqueue object", "obj", obj)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
func DeleteFunc(logger logr.Logger, enqueue EnqueueFunc) deleteFunc {
|
|
return func(obj interface{}) {
|
|
if err := enqueue(obj); err != nil {
|
|
logger.Error(err, "failed to enqueue object", "obj", obj)
|
|
}
|
|
}
|
|
}
|
|
|
|
func AddFuncT[K metav1.Object](logger logr.Logger, enqueue EnqueueFuncT[K]) addFuncT[K] {
|
|
return func(obj K) {
|
|
if err := enqueue(obj); err != nil {
|
|
logger.Error(err, "failed to enqueue object", "obj", obj)
|
|
}
|
|
}
|
|
}
|
|
|
|
func UpdateFuncT[K metav1.Object](logger logr.Logger, enqueue EnqueueFuncT[K]) updateFuncT[K] {
|
|
return func(old, obj K) {
|
|
if old.GetResourceVersion() != obj.GetResourceVersion() {
|
|
if err := enqueue(obj); err != nil {
|
|
logger.Error(err, "failed to enqueue object", "obj", obj)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
func DeleteFuncT[K metav1.Object](logger logr.Logger, enqueue EnqueueFuncT[K]) deleteFuncT[K] {
|
|
return func(obj K) {
|
|
if err := enqueue(obj); err != nil {
|
|
logger.Error(err, "failed to enqueue object", "obj", obj)
|
|
}
|
|
}
|
|
}
|