• Home
  • Features
  • Pricing
  • Docs
  • Announcements
  • Sign In

opendefensecloud / solution-arsenal / 27755794823

18 Jun 2026 11:18AM UTC coverage: 75.292% (-1.7%) from 77.003%
27755794823

Pull #621

github

web-flow
Merge 3e94622f3 into 69595b7d1
Pull Request #621: feat(controller): implement deletion protection via controller-managed finalizers

397 of 636 new or added lines in 7 files covered. (62.42%)

15 existing lines in 3 files now uncovered.

4001 of 5314 relevant lines covered (75.29%)

33.24 hits per line

Source File
Press 'n' to go to next uncovered line, 'b' for previous

71.7
/pkg/controller/profile_controller.go
1
// Copyright 2026 BWI GmbH and Solution Arsenal contributors
2
// SPDX-License-Identifier: Apache-2.0
3

4
package controller
5

6
import (
7
        "context"
8
        "fmt"
9
        "slices"
10

11
        corev1 "k8s.io/api/core/v1"
12
        apierrors "k8s.io/apimachinery/pkg/api/errors"
13
        metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
14
        "k8s.io/apimachinery/pkg/labels"
15
        "k8s.io/apimachinery/pkg/runtime"
16
        "k8s.io/apimachinery/pkg/types"
17
        "k8s.io/client-go/tools/events"
18
        ctrl "sigs.k8s.io/controller-runtime"
19
        "sigs.k8s.io/controller-runtime/pkg/client"
20
        "sigs.k8s.io/controller-runtime/pkg/handler"
21
        "sigs.k8s.io/controller-runtime/pkg/reconcile"
22

23
        solarv1alpha1 "go.opendefense.cloud/solar/api/solar/v1alpha1"
24
)
25

26
// bindingTargetKey returns a stable namespace/name key for a ReleaseBinding's target reference.
27
func bindingTargetKey(rb *solarv1alpha1.ReleaseBinding) string {
16✔
28
        ns := rb.Spec.TargetNamespace
16✔
29
        if ns == "" {
30✔
30
                ns = rb.Namespace
14✔
31
        }
14✔
32

33
        return ns + "/" + rb.Spec.TargetRef.Name
16✔
34
}
35

36
// targetKey returns the namespace/name key for a Target.
37
func targetKey(t *solarv1alpha1.Target) string {
23✔
38
        return t.Namespace + "/" + t.Name
23✔
39
}
23✔
40

41
// solarGroup is the API group for all solar resources.
42
const solarGroup = "solar.opendefense.cloud"
43

44
// grantPermits returns true if the ReferenceGrant allows a resource identified by
45
// (fromGroup, fromKind, fromNamespace) to reference a resource of (toGroup, toKind)
46
// in the grant's own namespace.
47
func grantPermits(grant *solarv1alpha1.ReferenceGrant, fromGroup, fromKind, fromNamespace, toGroup, toKind string) bool {
8✔
48
        hasFrom := false
8✔
49
        for _, f := range grant.Spec.From {
16✔
50
                if f.Namespace == fromNamespace && f.Kind == fromKind && f.Group == fromGroup {
16✔
51
                        hasFrom = true
8✔
52
                        break
8✔
53
                }
54
        }
55
        if !hasFrom {
8✔
UNCOV
56
                return false
×
UNCOV
57
        }
×
58
        for _, t := range grant.Spec.To {
16✔
59
                if t.Kind == toKind && t.Group == toGroup {
16✔
60
                        return true
8✔
61
                }
8✔
62
        }
63

64
        return false
×
65
}
66

67
// grantPermitsTargetAccess returns true if the ReferenceGrant allows a Profile in
68
// fromNamespace to reference Target resources in the grant's namespace.
69
func grantPermitsTargetAccess(grant *solarv1alpha1.ReferenceGrant, fromNamespace string) bool {
2✔
70
        return grantPermits(grant, solarGroup, "Profile", fromNamespace, solarGroup, "Target")
2✔
71
}
2✔
72

73
// grantsTargetResource returns true if the ReferenceGrant includes Target in its To list.
74
func grantsTargetResource(grant *solarv1alpha1.ReferenceGrant) bool {
52✔
75
        for _, t := range grant.Spec.To {
104✔
76
                if t.Kind == "Target" && t.Group == solarGroup {
100✔
77
                        return true
48✔
78
                }
48✔
79
        }
80

81
        return false
4✔
82
}
83

84
// grantPermitsComponentVersionAccess returns true if the ReferenceGrant allows a Release
85
// in fromNamespace to reference ComponentVersion resources in the grant's namespace.
86
func grantPermitsComponentVersionAccess(grant *solarv1alpha1.ReferenceGrant, fromNamespace string) bool {
6✔
87
        return grantPermits(grant, solarGroup, "Release", fromNamespace, solarGroup, "ComponentVersion")
6✔
88
}
6✔
89

90
// grantsComponentVersionResource returns true if the ReferenceGrant includes ComponentVersion in its To list.
91
func grantsComponentVersionResource(grant *solarv1alpha1.ReferenceGrant) bool {
25✔
92
        for _, t := range grant.Spec.To {
50✔
93
                if t.Kind == "ComponentVersion" && t.Group == solarGroup {
34✔
94
                        return true
9✔
95
                }
9✔
96
        }
97

98
        return false
16✔
99
}
100

101
// ProfileReconciler reconciles a Profile object.
102
// It evaluates the Profile's TargetSelector against all Targets in the namespace
103
// and creates/deletes ReleaseBindings accordingly.
104
// Cross-namespace Targets are included when a ReferenceGrant in the target's namespace
105
// grants the Profile's namespace access to "targets".
106
type ProfileReconciler struct {
107
        client.Client
108
        Scheme   *runtime.Scheme
109
        Recorder events.EventRecorder
110
        // WatchNamespace restricts reconciliation to this namespace.
111
        WatchNamespace string
112
}
113

114
//+kubebuilder:rbac:groups=solar.opendefense.cloud,resources=profiles,verbs=get;list;watch;update;patch
115
//+kubebuilder:rbac:groups=solar.opendefense.cloud,resources=profiles/status,verbs=get;update;patch
116
//+kubebuilder:rbac:groups=solar.opendefense.cloud,resources=profiles/finalizers,verbs=update
117
//+kubebuilder:rbac:groups=solar.opendefense.cloud,resources=releasebindings,verbs=get;list;watch;create;update;patch;delete
118
//+kubebuilder:rbac:groups=solar.opendefense.cloud,resources=releases,verbs=get;list;watch;update;patch
119
//+kubebuilder:rbac:groups=solar.opendefense.cloud,resources=releases/finalizers,verbs=update
120
//+kubebuilder:rbac:groups=solar.opendefense.cloud,resources=targets,verbs=get;list;watch
121
//+kubebuilder:rbac:groups=solar.opendefense.cloud,resources=referencegrants,verbs=get;list;watch
122
//+kubebuilder:rbac:groups=events.k8s.io,resources=events,verbs=create;patch
123

124
// Reconcile evaluates the Profile's TargetSelector and ensures matching ReleaseBindings exist.
125
// It also manages a deletion-protection finalizer on the referenced Release.
126
func (r *ProfileReconciler) Reconcile(ctx context.Context, req ctrl.Request) (ctrl.Result, error) {
88✔
127
        log := ctrl.LoggerFrom(ctx)
88✔
128

88✔
129
        log.V(1).Info("Profile is being reconciled", "req", req)
88✔
130

88✔
131
        if r.WatchNamespace != "" && req.Namespace != r.WatchNamespace {
132✔
132
                return ctrl.Result{}, nil
44✔
133
        }
44✔
134

135
        // Fetch Profile
136
        profile := &solarv1alpha1.Profile{}
44✔
137
        if err := r.Get(ctx, req.NamespacedName, profile); err != nil {
46✔
138
                if apierrors.IsNotFound(err) {
4✔
139
                        return ctrl.Result{}, nil
2✔
140
                }
2✔
141

142
                return ctrl.Result{}, errLogAndWrap(log, err, "failed to get Profile")
×
143
        }
144

145
        // Handle deletion.
146
        if !profile.DeletionTimestamp.IsZero() {
52✔
147
                // Block until all owned ReleaseBindings are fully gone from the API before unprotecting
10✔
148
                // the Release. This ensures the Release is never deletable while bindings still reference it.
10✔
149
                // The Owns() watch in SetupWithManager re-triggers this reconcile when each binding is removed.
10✔
150
                allBindings := &solarv1alpha1.ReleaseBindingList{}
10✔
151
                if err := r.List(ctx, allBindings, client.InNamespace(profile.Namespace)); err != nil {
10✔
NEW
152
                        return ctrl.Result{}, errLogAndWrap(log, err, "failed to list owned ReleaseBindings for deletion")
×
NEW
153
                }
×
154
                ownedBindings := &solarv1alpha1.ReleaseBindingList{}
10✔
155
                for i := range allBindings.Items {
16✔
156
                        if metav1.IsControlledBy(&allBindings.Items[i], profile) {
12✔
157
                                ownedBindings.Items = append(ownedBindings.Items, allBindings.Items[i])
6✔
158
                        }
6✔
159
                }
160

161
                var ownedExist bool
10✔
162
                for i := range ownedBindings.Items {
16✔
163
                        rb := &ownedBindings.Items[i]
6✔
164
                        ownedExist = true
6✔
165
                        if rb.DeletionTimestamp.IsZero() {
8✔
166
                                if err := r.Delete(ctx, rb); err != nil && !apierrors.IsNotFound(err) {
2✔
NEW
167
                                        return ctrl.Result{}, errLogAndWrap(log, err, "failed to delete owned ReleaseBinding during Profile deletion")
×
NEW
168
                                }
×
169
                        }
170
                }
171
                if ownedExist {
16✔
172
                        // Re-triggered by Owns() watch when the last binding is fully removed.
6✔
173
                        return ctrl.Result{}, nil
6✔
174
                }
6✔
175

176
                if profile.Spec.ReleaseRef.Name != "" {
8✔
177
                        release := &solarv1alpha1.Release{}
4✔
178
                        if err := r.Get(ctx, types.NamespacedName{Name: profile.Spec.ReleaseRef.Name, Namespace: profile.Namespace}, release); err != nil {
4✔
NEW
179
                                if !apierrors.IsNotFound(err) {
×
NEW
180
                                        return ctrl.Result{}, errLogAndWrap(log, err, "failed to get Release for finalizer cleanup")
×
NEW
181
                                }
×
182
                        } else if err := r.removeReleaseRefFinalizerIfUnreferenced(ctx, profile, release); err != nil {
4✔
NEW
183
                                return ctrl.Result{}, err
×
NEW
184
                        }
×
185
                }
186

187
                if slices.Contains(profile.Finalizers, profileFinalizer) {
8✔
188
                        latest := &solarv1alpha1.Profile{}
4✔
189
                        if err := r.Get(ctx, req.NamespacedName, latest); err != nil {
4✔
NEW
190
                                return ctrl.Result{}, errLogAndWrap(log, err, "failed to get latest Profile for finalizer removal")
×
NEW
191
                        }
×
192
                        original := latest.DeepCopy()
4✔
193
                        latest.Finalizers = slices.DeleteFunc(latest.Finalizers, func(s string) bool { return s == profileFinalizer })
8✔
194
                        if err := r.Patch(ctx, latest, client.MergeFrom(original)); err != nil {
4✔
NEW
195
                                return ctrl.Result{}, errLogAndWrap(log, err, "failed to remove finalizer from Profile")
×
NEW
196
                        }
×
197
                }
198

199
                return ctrl.Result{}, nil
4✔
200
        }
201

202
        // Ensure self-finalizer exists.
203
        if !slices.Contains(profile.Finalizers, profileFinalizer) {
44✔
204
                latest := &solarv1alpha1.Profile{}
12✔
205
                if err := r.Get(ctx, req.NamespacedName, latest); err != nil {
12✔
NEW
206
                        return ctrl.Result{}, errLogAndWrap(log, err, "failed to get latest Profile for finalizer addition")
×
NEW
207
                }
×
208
                original := latest.DeepCopy()
12✔
209
                latest.Finalizers = append(latest.Finalizers, profileFinalizer)
12✔
210
                if err := r.Patch(ctx, latest, client.MergeFrom(original)); err != nil {
12✔
NEW
211
                        return ctrl.Result{}, errLogAndWrap(log, err, "failed to add finalizer to Profile")
×
NEW
212
                }
×
213
        }
214

215
        // Protect the referenced Release from deletion.
216
        if profile.Spec.ReleaseRef.Name != "" {
64✔
217
                release := &solarv1alpha1.Release{}
32✔
218
                if err := r.Get(ctx, types.NamespacedName{Name: profile.Spec.ReleaseRef.Name, Namespace: profile.Namespace}, release); err != nil {
50✔
219
                        if !apierrors.IsNotFound(err) {
18✔
NEW
220
                                return ctrl.Result{}, errLogAndWrap(log, err, "failed to get Release for protection finalizer")
×
NEW
221
                        }
×
222
                } else if !slices.Contains(release.Finalizers, releaseRefFinalizer) {
19✔
223
                        latest := release.DeepCopy()
5✔
224
                        latest.Finalizers = append(latest.Finalizers, releaseRefFinalizer)
5✔
225
                        if err := r.Patch(ctx, latest, client.MergeFrom(release)); err != nil {
5✔
NEW
226
                                return ctrl.Result{}, errLogAndWrap(log, err, "failed to add protection finalizer to Release")
×
NEW
227
                        }
×
228
                }
229
        }
230

231
        // Evaluate TargetSelector against all Targets
232
        selector, err := metav1.LabelSelectorAsSelector(&profile.Spec.TargetSelector)
32✔
233
        if err != nil {
32✔
234
                log.Error(err, "invalid targetSelector in Profile")
×
235

×
236
                return ctrl.Result{}, nil
×
237
        }
×
238

239
        // Collect matching targets from the profile's own namespace
240
        sameNsTargets := &solarv1alpha1.TargetList{}
32✔
241
        if err := r.List(ctx, sameNsTargets,
32✔
242
                client.InNamespace(profile.Namespace),
32✔
243
                client.MatchingLabelsSelector{Selector: selector},
32✔
244
        ); err != nil {
32✔
245
                return ctrl.Result{}, errLogAndWrap(log, err, "failed to list Targets")
×
246
        }
×
247

248
        allTargets := make([]solarv1alpha1.Target, 0, len(sameNsTargets.Items))
32✔
249
        allTargets = append(allTargets, sameNsTargets.Items...)
32✔
250

32✔
251
        // Collect cross-namespace targets via ReferenceGrants.
32✔
252
        // A ReferenceGrant in namespace B listing the profile's namespace in From and
32✔
253
        // "targets" in To allows this Profile to select Targets from namespace B.
32✔
254
        //
32✔
255
        // FIXME: listing all ReferenceGrants cluster-wide on every reconcile is a cache
32✔
256
        // scan only (no etcd round-trip, no host-cluster impact), but may become a
32✔
257
        // bottleneck at scale. Consider adding a field index on ReferenceGrants keyed by
32✔
258
        // the namespaces they grant access to so we can filter server-side.
32✔
259
        grantList := &solarv1alpha1.ReferenceGrantList{}
32✔
260
        if err := r.List(ctx, grantList); err != nil {
32✔
261
                return ctrl.Result{}, errLogAndWrap(log, err, "failed to list ReferenceGrants")
×
262
        }
×
263

264
        for i := range grantList.Items {
34✔
265
                grant := &grantList.Items[i]
2✔
266
                if grant.Namespace == profile.Namespace {
2✔
267
                        // same-namespace targets already covered above
×
268
                        continue
×
269
                }
270
                if !grantPermitsTargetAccess(grant, profile.Namespace) {
2✔
UNCOV
271
                        continue
×
272
                }
273
                crossNsTargets := &solarv1alpha1.TargetList{}
2✔
274
                if err := r.List(ctx, crossNsTargets,
2✔
275
                        client.InNamespace(grant.Namespace),
2✔
276
                        client.MatchingLabelsSelector{Selector: selector},
2✔
277
                ); err != nil {
2✔
278
                        return ctrl.Result{}, errLogAndWrap(log, err, "failed to list cross-namespace Targets in "+grant.Namespace)
×
279
                }
×
280
                allTargets = append(allTargets, crossNsTargets.Items...)
2✔
281
        }
282

283
        // Build set of desired ReleaseBindings (one per matching target, keyed by namespace/name)
284
        desiredTargets := map[string]solarv1alpha1.Target{}
32✔
285
        for _, t := range allTargets {
55✔
286
                desiredTargets[targetKey(&t)] = t
23✔
287
        }
23✔
288

289
        // List existing ReleaseBindings owned by this Profile
290
        allBindings := &solarv1alpha1.ReleaseBindingList{}
32✔
291
        if err := r.List(ctx, allBindings, client.InNamespace(profile.Namespace)); err != nil {
32✔
NEW
292
                return ctrl.Result{}, errLogAndWrap(log, err, "failed to list ReleaseBindings")
×
NEW
293
        }
×
294
        existingBindings := &solarv1alpha1.ReleaseBindingList{}
32✔
295
        for i := range allBindings.Items {
48✔
296
                if metav1.IsControlledBy(&allBindings.Items[i], profile) {
32✔
297
                        existingBindings.Items = append(existingBindings.Items, allBindings.Items[i])
16✔
298
                }
16✔
299
        }
300

301
        // Delete ReleaseBindings for targets that no longer match
302
        existingByKey := map[string]*solarv1alpha1.ReleaseBinding{}
32✔
303
        for i := range existingBindings.Items {
48✔
304
                rb := &existingBindings.Items[i]
16✔
305
                key := bindingTargetKey(rb)
16✔
306
                existingByKey[key] = rb
16✔
307

16✔
308
                if _, desired := desiredTargets[key]; !desired {
19✔
309
                        log.V(1).Info("Deleting ReleaseBinding for unmatched target", "key", key)
3✔
310
                        if err := r.Delete(ctx, rb); err != nil && !apierrors.IsNotFound(err) {
3✔
311
                                return ctrl.Result{}, errLogAndWrap(log, err, "failed to delete ReleaseBinding")
×
312
                        }
×
313

314
                        r.Recorder.Eventf(profile, nil, corev1.EventTypeNormal, "Deleted", "Delete",
3✔
315
                                "Deleted ReleaseBinding for target %s", key)
3✔
316
                }
317
        }
318

319
        // Create ReleaseBindings for new matching targets
320
        for key, target := range desiredTargets {
55✔
321
                if _, exists := existingByKey[key]; exists {
36✔
322
                        continue
13✔
323
                }
324

325
                crossNs := ""
10✔
326
                if target.Namespace != profile.Namespace {
12✔
327
                        crossNs = target.Namespace
2✔
328
                }
2✔
329

330
                rb := &solarv1alpha1.ReleaseBinding{
10✔
331
                        ObjectMeta: metav1.ObjectMeta{
10✔
332
                                // We need to truncated the name: 57 (input) + 1 (-) + 5 (appended by generated) = 63 (max chars allowed)
10✔
333
                                GenerateName: truncateName(fmt.Sprintf("%s-%s", profile.Name, target.Name), 57) + "-",
10✔
334
                                Namespace:    profile.Namespace,
10✔
335
                        },
10✔
336
                        Spec: solarv1alpha1.ReleaseBindingSpec{
10✔
337
                                TargetRef:       corev1.LocalObjectReference{Name: target.Name},
10✔
338
                                TargetNamespace: crossNs,
10✔
339
                                ReleaseRef:      profile.Spec.ReleaseRef,
10✔
340
                        },
10✔
341
                }
10✔
342
                if err := ctrl.SetControllerReference(profile, rb, r.Scheme); err != nil {
10✔
343
                        return ctrl.Result{}, errLogAndWrap(log, err, "failed to set controller reference on ReleaseBinding")
×
344
                }
×
345

346
                if err := r.Create(ctx, rb); err != nil {
10✔
347
                        if apierrors.IsAlreadyExists(err) {
×
348
                                continue
×
349
                        }
350

351
                        return ctrl.Result{}, errLogAndWrap(log, err, "failed to create ReleaseBinding")
×
352
                }
353

354
                log.V(1).Info("Created ReleaseBinding for target", "key", key)
10✔
355
                r.Recorder.Eventf(profile, nil, corev1.EventTypeNormal, "Created", "Create",
10✔
356
                        "Created ReleaseBinding for target %s", key)
10✔
357
        }
358

359
        // Update status
360
        original := profile.DeepCopy()
32✔
361
        profile.Status.MatchedTargets = len(desiredTargets)
32✔
362
        if profile.Status.MatchedTargets != original.Status.MatchedTargets {
47✔
363
                if err := r.Status().Update(ctx, profile); err != nil {
24✔
364
                        return ctrl.Result{}, errLogAndWrap(log, err, "failed to update Profile status")
9✔
365
                }
9✔
366
        }
367

368
        return ctrl.Result{}, nil
23✔
369
}
370

371
// SetupWithManager sets up the controller with the Manager.
372
func (r *ProfileReconciler) SetupWithManager(mgr ctrl.Manager) error {
1✔
373
        return ctrl.NewControllerManagedBy(mgr).
1✔
374
                For(&solarv1alpha1.Profile{}).
1✔
375
                Owns(&solarv1alpha1.ReleaseBinding{}).
1✔
376
                Watches(
1✔
377
                        &solarv1alpha1.Target{},
1✔
378
                        handler.EnqueueRequestsFromMapFunc(r.mapTargetToProfiles),
1✔
379
                ).
1✔
380
                Watches(
1✔
381
                        &solarv1alpha1.ReferenceGrant{},
1✔
382
                        handler.EnqueueRequestsFromMapFunc(r.mapReferenceGrantToProfiles),
1✔
383
                ).
1✔
384
                Complete(r)
1✔
385
}
1✔
386

387
// mapTargetToProfiles maps a Target to all Profiles that might match it,
388
// including Profiles in namespaces that have been granted access via ReferenceGrant.
389
func (r *ProfileReconciler) mapTargetToProfiles(ctx context.Context, obj client.Object) []reconcile.Request {
413✔
390
        log := ctrl.LoggerFrom(ctx)
413✔
391

413✔
392
        target, ok := obj.(*solarv1alpha1.Target)
413✔
393
        if !ok {
413✔
394
                return nil
×
395
        }
×
396

397
        targetLabels := labels.Set(target.Labels)
413✔
398
        var requests []reconcile.Request
413✔
399

413✔
400
        // Enqueue profiles in the target's own namespace
413✔
401
        profileList := &solarv1alpha1.ProfileList{}
413✔
402
        if err := r.List(ctx, profileList, client.InNamespace(target.Namespace)); err != nil {
413✔
403
                log.Error(err, "failed to list Profiles for Target mapping")
×
404

×
405
                return nil
×
406
        }
×
407

408
        for _, profile := range profileList.Items {
464✔
409
                selector, err := metav1.LabelSelectorAsSelector(&profile.Spec.TargetSelector)
51✔
410
                if err != nil {
51✔
411
                        continue
×
412
                }
413

414
                if selector.Matches(targetLabels) {
93✔
415
                        requests = append(requests, reconcile.Request{
42✔
416
                                NamespacedName: client.ObjectKeyFromObject(&profile),
42✔
417
                        })
42✔
418
                }
42✔
419
        }
420

421
        // Enqueue profiles in namespaces that have been granted access to targets in
422
        // this target's namespace via a ReferenceGrant.
423
        grantList := &solarv1alpha1.ReferenceGrantList{}
413✔
424
        if err := r.List(ctx, grantList, client.InNamespace(target.Namespace)); err != nil {
413✔
425
                log.Error(err, "failed to list ReferenceGrants for cross-namespace Target mapping")
×
426

×
427
                return requests
×
428
        }
×
429

430
        for i := range grantList.Items {
453✔
431
                grant := &grantList.Items[i]
40✔
432
                if !grantsTargetResource(grant) {
40✔
433
                        continue
×
434
                }
435
                for _, from := range grant.Spec.From {
80✔
436
                        if from.Namespace == target.Namespace || from.Kind != "Profile" {
80✔
437
                                continue
40✔
438
                        }
439
                        fromProfiles := &solarv1alpha1.ProfileList{}
×
440
                        if err := r.List(ctx, fromProfiles, client.InNamespace(from.Namespace)); err != nil {
×
441
                                log.Error(err, "failed to list Profiles in granted namespace", "namespace", from.Namespace)
×
442
                                continue
×
443
                        }
444
                        for _, p := range fromProfiles.Items {
×
445
                                selector, err := metav1.LabelSelectorAsSelector(&p.Spec.TargetSelector)
×
446
                                if err != nil {
×
447
                                        continue
×
448
                                }
449
                                if selector.Matches(targetLabels) {
×
450
                                        requests = append(requests, reconcile.Request{
×
451
                                                NamespacedName: client.ObjectKeyFromObject(&p),
×
452
                                        })
×
453
                                }
×
454
                        }
455
                }
456
        }
457

458
        return requests
413✔
459
}
460

461
// mapReferenceGrantToProfiles enqueues all Profiles in the namespaces listed in
462
// a ReferenceGrant's From field, allowing them to re-evaluate cross-namespace matches.
463
func (r *ProfileReconciler) mapReferenceGrantToProfiles(ctx context.Context, obj client.Object) []reconcile.Request {
12✔
464
        log := ctrl.LoggerFrom(ctx)
12✔
465

12✔
466
        grant, ok := obj.(*solarv1alpha1.ReferenceGrant)
12✔
467
        if !ok {
12✔
468
                return nil
×
469
        }
×
470

471
        if !grantsTargetResource(grant) {
16✔
472
                return nil
4✔
473
        }
4✔
474

475
        var requests []reconcile.Request
8✔
476
        for _, from := range grant.Spec.From {
16✔
477
                if from.Kind != "Profile" || from.Group != solarGroup {
12✔
478
                        continue
4✔
479
                }
480
                profiles := &solarv1alpha1.ProfileList{}
4✔
481
                if err := r.List(ctx, profiles, client.InNamespace(from.Namespace)); err != nil {
4✔
482
                        log.Error(err, "failed to list Profiles for ReferenceGrant mapping", "namespace", from.Namespace)
×
483
                        continue
×
484
                }
485
                for _, p := range profiles.Items {
5✔
486
                        requests = append(requests, reconcile.Request{
1✔
487
                                NamespacedName: client.ObjectKeyFromObject(&p),
1✔
488
                        })
1✔
489
                }
1✔
490
        }
491

492
        return requests
8✔
493
}
494

495
// removeReleaseRefFinalizerIfUnreferenced removes releaseRefFinalizer from release when no active
496
// Profile or ReleaseBinding (excluding the deleting Profile and its owned ReleaseBindings) still
497
// references it.
498
func (r *ProfileReconciler) removeReleaseRefFinalizerIfUnreferenced(ctx context.Context, deletingProfile *solarv1alpha1.Profile, release *solarv1alpha1.Release) error {
4✔
499
        if !slices.Contains(release.Finalizers, releaseRefFinalizer) {
4✔
NEW
500
                return nil
×
NEW
501
        }
×
502

503
        // Count Profiles (excluding self) referencing this Release.
504
        profileList := &solarv1alpha1.ProfileList{}
4✔
505
        if err := r.List(ctx, profileList,
4✔
506
                client.InNamespace(release.Namespace),
4✔
507
                client.MatchingFields{indexProfileByReleaseName: release.Name},
4✔
508
        ); err != nil {
4✔
NEW
509
                return errLogAndWrap(ctrl.LoggerFrom(ctx), err, "failed to list Profiles for Release finalizer check")
×
NEW
510
        }
×
511

512
        for _, p := range profileList.Items {
8✔
513
                if p.Name == deletingProfile.Name {
8✔
514
                        continue
4✔
515
                }
NEW
516
                if !p.DeletionTimestamp.IsZero() {
×
NEW
517
                        continue
×
518
                }
519

NEW
520
                return nil
×
521
        }
522

523
        // Count ReleaseBindings (excluding those owned by the deleting Profile) referencing this Release.
524
        bindingList := &solarv1alpha1.ReleaseBindingList{}
4✔
525
        if err := r.List(ctx, bindingList,
4✔
526
                client.InNamespace(release.Namespace),
4✔
527
                client.MatchingFields{indexReleaseBindingReleaseName: release.Name},
4✔
528
        ); err != nil {
4✔
NEW
529
                return errLogAndWrap(ctrl.LoggerFrom(ctx), err, "failed to list ReleaseBindings for Release finalizer check")
×
NEW
530
        }
×
531

532
        for _, rb := range bindingList.Items {
4✔
NEW
533
                if metav1.IsControlledBy(&rb, deletingProfile) {
×
NEW
534
                        continue // owned by the Profile being deleted; K8s GC will remove these
×
535
                }
NEW
536
                if !rb.DeletionTimestamp.IsZero() {
×
NEW
537
                        continue
×
538
                }
539

540
                // Check if this binding's owner Profile is also being deleted (concurrent deletion).
NEW
541
                ownerRef := metav1.GetControllerOf(&rb)
×
NEW
542
                if ownerRef != nil && ownerRef.Kind == "Profile" && ownerRef.APIVersion == solarv1alpha1.SchemeGroupVersion.String() {
×
NEW
543
                        ownerProfile := &solarv1alpha1.Profile{}
×
NEW
544
                        err := r.Get(ctx, types.NamespacedName{Name: ownerRef.Name, Namespace: rb.Namespace}, ownerProfile)
×
NEW
545
                        if apierrors.IsNotFound(err) || (err == nil && !ownerProfile.DeletionTimestamp.IsZero()) {
×
NEW
546
                                continue // owner Profile is gone or being deleted, this binding will be GC'd
×
547
                        }
NEW
548
                        if err != nil {
×
NEW
549
                                return errLogAndWrap(ctrl.LoggerFrom(ctx), err, "failed to check owner Profile for concurrent deletion")
×
NEW
550
                        }
×
551
                }
552

NEW
553
                return nil
×
554
        }
555

556
        freshRelease := &solarv1alpha1.Release{}
4✔
557
        if err := r.Get(ctx, client.ObjectKeyFromObject(release), freshRelease); err != nil {
4✔
NEW
558
                if apierrors.IsNotFound(err) {
×
NEW
559
                        return nil
×
NEW
560
                }
×
561

NEW
562
                return errLogAndWrap(ctrl.LoggerFrom(ctx), err, "failed to get latest Release for finalizer removal")
×
563
        }
564
        original := freshRelease.DeepCopy()
4✔
565
        freshRelease.Finalizers = slices.DeleteFunc(freshRelease.Finalizers, func(s string) bool { return s == releaseRefFinalizer })
11✔
566
        if err := r.Patch(ctx, freshRelease, client.MergeFrom(original)); err != nil {
4✔
NEW
567
                return errLogAndWrap(ctrl.LoggerFrom(ctx), err, "failed to remove protection finalizer from Release")
×
NEW
568
        }
×
569

570
        ctrl.LoggerFrom(ctx).V(1).Info("Removed protection finalizer from Release", "release", release.Name)
4✔
571

4✔
572
        return nil
4✔
573
}
STATUS · Troubleshooting · Open an Issue · Sales · Support · CAREERS · ENTERPRISE · START FREE · SCHEDULE DEMO
ANNOUNCEMENTS · TWITTER · TOS & SLA · Supported CI Services · What's a CI service? · Automated Testing

© 2026 Coveralls, Inc