mirror of
https://github.com/projectcapsule/capsule.git
synced 2026-04-07 11:17:30 +00:00
* fix(controller): decode old object for delete requests Signed-off-by: Oliver Bähler <oliverbaehler@hotmail.com> * chore: modernize golang Signed-off-by: Oliver Bähler <oliverbaehler@hotmail.com> * chore: modernize golang Signed-off-by: Oliver Bähler <oliverbaehler@hotmail.com> * chore: modernize golang Signed-off-by: Oliver Bähler <oliverbaehler@hotmail.com> * fix(config): remove usergroups default Signed-off-by: Oliver Bähler <oliverbaehler@hotmail.com> * fix(config): remove usergroups default Signed-off-by: Oliver Bähler <oliverbaehler@hotmail.com> * sec(ghsa-2ww6-hf35-mfjm): intercept namespace subresource Signed-off-by: Oliver Bähler <oliverbaehler@hotmail.com> * feat(api): add rulestatus api Signed-off-by: Oliver Bähler <oliverbaehler@hotmail.com> * chore: conflicts Signed-off-by: Oliver Bähler <oliverbaehler@hotmail.com> * chore: conflicts Signed-off-by: Oliver Bähler <oliverbaehler@hotmail.com> * chore: conflicts Signed-off-by: Oliver Bähler <oliverbaehler@hotmail.com> * chore: conflicts Signed-off-by: Oliver Bähler <oliverbaehler@hotmail.com> * chore: conflicts Signed-off-by: Oliver Bähler <oliverbaehler@hotmail.com> * chore: conflicts Signed-off-by: Oliver Bähler <oliverbaehler@hotmail.com> * chore: conflicts Signed-off-by: Oliver Bähler <oliverbaehler@hotmail.com> * chore: conflicts Signed-off-by: Oliver Bähler <oliverbaehler@hotmail.com> * chore: conflicts Signed-off-by: Oliver Bähler <oliverbaehler@hotmail.com> * chore: conflicts Signed-off-by: Oliver Bähler <oliverbaehler@hotmail.com> * chore: conflicts Signed-off-by: Oliver Bähler <oliverbaehler@hotmail.com> * feat(api): add rulestatus api Signed-off-by: Oliver Bähler <oliverbaehler@hotmail.com> * feat(api): add rulestatus api Signed-off-by: Oliver Bähler <oliverbaehler@hotmail.com> * feat(api): add rulestatus api Signed-off-by: Oliver Bähler <oliverbaehler@hotmail.com> * feat(api): add rulestatus api Signed-off-by: Oliver Bähler <oliverbaehler@hotmail.com> * feat(api): add rulestatus api Signed-off-by: Oliver Bähler <oliverbaehler@hotmail.com> * feat(api): add rulestatus api Signed-off-by: Oliver Bähler <oliverbaehler@hotmail.com> --------- Signed-off-by: Oliver Bähler <oliverbaehler@hotmail.com>
183 lines
4.5 KiB
Go
183 lines
4.5 KiB
Go
// Copyright 2020-2026 Project Capsule Authors
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
package tenant
|
|
|
|
import (
|
|
"context"
|
|
|
|
"k8s.io/apimachinery/pkg/labels"
|
|
"k8s.io/apimachinery/pkg/selection"
|
|
"k8s.io/apimachinery/pkg/types"
|
|
"k8s.io/client-go/util/retry"
|
|
"k8s.io/client-go/util/workqueue"
|
|
"sigs.k8s.io/controller-runtime/pkg/client"
|
|
"sigs.k8s.io/controller-runtime/pkg/event"
|
|
"sigs.k8s.io/controller-runtime/pkg/handler"
|
|
"sigs.k8s.io/controller-runtime/pkg/reconcile"
|
|
|
|
capsulev1beta2 "github.com/projectcapsule/capsule/api/v1beta2"
|
|
"github.com/projectcapsule/capsule/pkg/utils"
|
|
)
|
|
|
|
func (r *Manager) statusOnlyHandlerClasses(
|
|
fn func(ctx context.Context, perTenant func(context.Context, *capsulev1beta2.Tenant) error) error,
|
|
perTenant func(context.Context, *capsulev1beta2.Tenant) error,
|
|
errMsg string,
|
|
) *handler.TypedFuncs[client.Object, reconcile.Request] {
|
|
return &handler.TypedFuncs[client.Object, reconcile.Request]{
|
|
CreateFunc: func(
|
|
ctx context.Context,
|
|
_ event.TypedCreateEvent[client.Object],
|
|
_ workqueue.TypedRateLimitingInterface[reconcile.Request],
|
|
) {
|
|
if err := fn(ctx, perTenant); err != nil {
|
|
r.Log.Error(err, errMsg)
|
|
}
|
|
},
|
|
UpdateFunc: func(
|
|
ctx context.Context,
|
|
_ event.TypedUpdateEvent[client.Object],
|
|
_ workqueue.TypedRateLimitingInterface[reconcile.Request],
|
|
) {
|
|
if err := fn(ctx, perTenant); err != nil {
|
|
r.Log.Error(err, errMsg)
|
|
}
|
|
},
|
|
DeleteFunc: func(
|
|
ctx context.Context,
|
|
_ event.TypedDeleteEvent[client.Object],
|
|
_ workqueue.TypedRateLimitingInterface[reconcile.Request],
|
|
) {
|
|
if err := fn(ctx, perTenant); err != nil {
|
|
r.Log.Error(err, errMsg)
|
|
}
|
|
},
|
|
}
|
|
}
|
|
|
|
func (r *Manager) enqueueTenantsForTenantOwner(
|
|
ctx context.Context,
|
|
tenantOwner client.Object,
|
|
q workqueue.TypedRateLimitingInterface[reconcile.Request],
|
|
) {
|
|
var tenants capsulev1beta2.TenantList
|
|
if err := r.List(ctx, &tenants); err != nil {
|
|
r.Log.Error(err, "failed to list Tenants for Tenant Owner event")
|
|
|
|
return
|
|
}
|
|
|
|
owner, ok := tenantOwner.(*capsulev1beta2.TenantOwner)
|
|
if !ok {
|
|
return
|
|
}
|
|
|
|
for i := range tenants.Items {
|
|
tnt := &tenants.Items[i]
|
|
|
|
if _, found := tnt.Status.Owners.FindOwner(
|
|
owner.Spec.Name,
|
|
owner.Spec.Kind,
|
|
); !found {
|
|
continue
|
|
}
|
|
|
|
q.Add(reconcile.Request{
|
|
NamespacedName: types.NamespacedName{
|
|
Name: tnt.Name,
|
|
},
|
|
})
|
|
}
|
|
}
|
|
|
|
func (r *Manager) enqueueForTenantsWithCondition(
|
|
ctx context.Context,
|
|
obj client.Object,
|
|
q workqueue.TypedRateLimitingInterface[reconcile.Request],
|
|
fn func(*capsulev1beta2.Tenant, client.Object) bool,
|
|
) {
|
|
var tenants capsulev1beta2.TenantList
|
|
if err := r.List(ctx, &tenants); err != nil {
|
|
r.Log.Error(err, "failed to list Tenants for class event")
|
|
|
|
return
|
|
}
|
|
|
|
for i := range tenants.Items {
|
|
tnt := &tenants.Items[i]
|
|
|
|
if !fn(tnt, obj) {
|
|
continue
|
|
}
|
|
|
|
q.Add(reconcile.Request{
|
|
NamespacedName: types.NamespacedName{
|
|
Name: tnt.Name,
|
|
},
|
|
})
|
|
}
|
|
}
|
|
|
|
func (r *Manager) enqueueAllTenants(ctx context.Context, _ client.Object) []reconcile.Request {
|
|
var tenants capsulev1beta2.TenantList
|
|
if err := r.List(ctx, &tenants); err != nil {
|
|
r.Log.Error(err, "failed to list Tenants for class event")
|
|
|
|
return nil
|
|
}
|
|
|
|
reqs := make([]reconcile.Request, 0, len(tenants.Items))
|
|
for i := range tenants.Items {
|
|
reqs = append(reqs, reconcile.Request{
|
|
NamespacedName: types.NamespacedName{
|
|
Name: tenants.Items[i].Name,
|
|
},
|
|
})
|
|
}
|
|
|
|
return reqs
|
|
}
|
|
|
|
// pruningResources is taking care of removing the no more requested sub-resources as LimitRange, ResourceQuota or
|
|
// NetworkPolicy using the "exists" and "notin" LabelSelector to perform an outer-join removal.
|
|
func (r *Manager) pruningResources(ctx context.Context, ns string, keys []string, obj client.Object) (err error) {
|
|
var capsuleLabel string
|
|
|
|
if capsuleLabel, err = utils.GetTypeLabel(obj); err != nil {
|
|
return err
|
|
}
|
|
|
|
selector := labels.NewSelector()
|
|
|
|
var exists *labels.Requirement
|
|
|
|
if exists, err = labels.NewRequirement(capsuleLabel, selection.Exists, []string{}); err != nil {
|
|
return err
|
|
}
|
|
|
|
selector = selector.Add(*exists)
|
|
|
|
if len(keys) > 0 {
|
|
var notIn *labels.Requirement
|
|
|
|
if notIn, err = labels.NewRequirement(capsuleLabel, selection.NotIn, keys); err != nil {
|
|
return err
|
|
}
|
|
|
|
selector = selector.Add(*notIn)
|
|
}
|
|
|
|
r.Log.V(4).Info("pruning objects with label selector " + selector.String())
|
|
|
|
return retry.RetryOnConflict(retry.DefaultBackoff, func() error {
|
|
return r.DeleteAllOf(ctx, obj, &client.DeleteAllOfOptions{
|
|
ListOptions: client.ListOptions{
|
|
LabelSelector: selector,
|
|
Namespace: ns,
|
|
},
|
|
DeleteOptions: client.DeleteOptions{},
|
|
})
|
|
})
|
|
}
|