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

opendefensecloud / solution-arsenal / 27684070383

17 Jun 2026 10:58AM UTC coverage: 76.728%. First build
27684070383

Pull #611

github

web-flow
Merge 1bd3e1ac4 into 6a08235ee
Pull Request #611: fix: enable tests

44 of 51 new or added lines in 2 files covered. (86.27%)

3640 of 4744 relevant lines covered (76.73%)

43.52 hits per line

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

72.42
/pkg/controller/target_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
        "errors"
9
        "fmt"
10
        "net/url"
11
        "slices"
12
        "sort"
13
        "strings"
14
        "time"
15

16
        ociname "github.com/google/go-containerregistry/pkg/name"
17
        corev1 "k8s.io/api/core/v1"
18
        apiequality "k8s.io/apimachinery/pkg/api/equality"
19
        apierrors "k8s.io/apimachinery/pkg/api/errors"
20
        apimeta "k8s.io/apimachinery/pkg/api/meta"
21
        metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
22
        "k8s.io/apimachinery/pkg/labels"
23
        "k8s.io/apimachinery/pkg/runtime"
24
        "k8s.io/apimachinery/pkg/types"
25
        "k8s.io/client-go/tools/events"
26
        ctrl "sigs.k8s.io/controller-runtime"
27
        "sigs.k8s.io/controller-runtime/pkg/builder"
28
        "sigs.k8s.io/controller-runtime/pkg/client"
29
        "sigs.k8s.io/controller-runtime/pkg/handler"
30
        "sigs.k8s.io/controller-runtime/pkg/reconcile"
31

32
        solarv1alpha1 "go.opendefense.cloud/solar/api/solar/v1alpha1"
33
)
34

35
const (
36
        targetFinalizer = "solar.opendefense.cloud/target-finalizer"
37

38
        ConditionTypeRegistryResolved = "RegistryResolved"
39
        ConditionTypeReleasesResolved = "ReleasesResolved"
40
        ConditionTypeReleasesRendered = "ReleasesRendered"
41
        ConditionTypeBootstrapReady   = "BootstrapReady"
42
)
43

44
var ErrReleaseNotRenderedYet = errors.New("release is not rendered yet")
45

46
type releaseInfo struct {
47
        // bindingKey is "<namespace>/<name>" of the originating ReleaseBinding, used as a
48
        // deterministic tiebreaker when two releases share the same priority.
49
        bindingKey string
50
        name       string
51
        // uniqueName is the deduplication key computed by resolveReleaseConflicts:
52
        // Spec.UniqueName when set, otherwise the parent Component name from the CV.
53
        // It is guaranteed unique across all surviving releases and used as the
54
        // bootstrap map key to avoid collisions between same-named cross-namespace releases.
55
        uniqueName          string
56
        release             *solarv1alpha1.Release
57
        cv                  *solarv1alpha1.ComponentVersion
58
        rtName              string
59
        chartURL            string
60
        artifactName        string
61
        artifactBindingName string
62
}
63

64
type TargetReconciler struct {
65
        client.Client
66
        Scheme    *runtime.Scheme
67
        Recorder  events.EventRecorder
68
        APIReader client.Reader
69
        // WatchNamespace restricts reconciliation to this namespace.
70
        // Should be empty in production (watches all namespaces).
71
        // Intended for use in integration tests only.
72
        WatchNamespace string
73
        // RegistryBindingStrict enables strict registry binding mode.
74
        // When true, rendering fails if a resource's registry host has no
75
        // matching RegistryBinding. When false (default/relaxed), unmatched
76
        // hosts are treated as anonymous pull (no secretRef rendered).
77
        RegistryBindingStrict bool
78
}
79

80
//+kubebuilder:rbac:groups=solar.opendefense.cloud,resources=targets,verbs=get;list;watch;create;update;patch;delete
81
//+kubebuilder:rbac:groups=solar.opendefense.cloud,resources=targets/status,verbs=get;update;patch
82
//+kubebuilder:rbac:groups=solar.opendefense.cloud,resources=targets/finalizers,verbs=update
83
//+kubebuilder:rbac:groups=solar.opendefense.cloud,resources=registries,verbs=get;list;watch
84
//+kubebuilder:rbac:groups=solar.opendefense.cloud,resources=releasebindings,verbs=get;list;watch
85
//+kubebuilder:rbac:groups=solar.opendefense.cloud,resources=registrybindings,verbs=get;list;watch
86
//+kubebuilder:rbac:groups=solar.opendefense.cloud,resources=releases,verbs=get;list;watch
87
//+kubebuilder:rbac:groups=solar.opendefense.cloud,resources=componentversions,verbs=get;list;watch
88
//+kubebuilder:rbac:groups=solar.opendefense.cloud,resources=referencegrants,verbs=get;list;watch
89
//+kubebuilder:rbac:groups=solar.opendefense.cloud,resources=rendertasks,verbs=get;list;watch;create;update;patch;delete
90
//+kubebuilder:rbac:groups=solar.opendefense.cloud,resources=renderartifacts,verbs=get;list;watch;create;update;patch
91
//+kubebuilder:rbac:groups=solar.opendefense.cloud,resources=renderbindings,verbs=get;list;watch;create;update;patch;delete
92
//+kubebuilder:rbac:groups="",resources=secrets,verbs=get
93
//+kubebuilder:rbac:groups=events.k8s.io,resources=events,verbs=create;patch
94

95
// Reconcile collects ReleaseBindings, resolves the render registry, creates per-release
96
// RenderTasks (with dedup), and creates a per-target bootstrap RenderTask.
97
func (r *TargetReconciler) Reconcile(ctx context.Context, req ctrl.Request) (ctrl.Result, error) {
278✔
98
        log := ctrl.LoggerFrom(ctx)
278✔
99

278✔
100
        log.V(1).Info("Target is being reconciled", "req", req)
278✔
101

278✔
102
        if r.WatchNamespace != "" && req.Namespace != r.WatchNamespace {
370✔
103
                return ctrl.Result{}, nil
92✔
104
        }
92✔
105

106
        // Fetch target
107
        target := &solarv1alpha1.Target{}
186✔
108
        if err := r.Get(ctx, req.NamespacedName, target); err != nil {
191✔
109
                if apierrors.IsNotFound(err) {
10✔
110
                        return ctrl.Result{}, nil
5✔
111
                }
5✔
112

113
                return ctrl.Result{}, errLogAndWrap(log, err, "failed to get object")
×
114
        }
115

116
        // Handle deletion
117
        if !target.DeletionTimestamp.IsZero() {
183✔
118
                log.V(1).Info("Target is being deleted")
2✔
119
                r.Recorder.Eventf(target, nil, corev1.EventTypeWarning, "Deleting", "Reconcile", "Target is being deleted, cleaning up RenderTasks")
2✔
120

2✔
121
                // Delete owned RenderTasks
2✔
122
                if err := r.deleteOwnedRenderTasks(ctx, target); err != nil {
2✔
123
                        return ctrl.Result{}, errLogAndWrap(log, err, "failed to delete owned RenderTasks")
×
124
                }
×
125

126
                // Delete owned RenderBindings so the GC controller can clean up orphaned RenderArtifacts.
127
                if err := r.deleteOwnedRenderBindings(ctx, target); err != nil {
2✔
128
                        return ctrl.Result{}, errLogAndWrap(log, err, "failed to delete owned RenderBindings")
×
129
                }
×
130

131
                // Remove finalizer
132
                if slices.Contains(target.Finalizers, targetFinalizer) {
4✔
133
                        latest := &solarv1alpha1.Target{}
2✔
134
                        if err := r.Get(ctx, req.NamespacedName, latest); err != nil {
2✔
135
                                return ctrl.Result{}, errLogAndWrap(log, err, "failed to get latest Target for finalizer removal")
×
136
                        }
×
137

138
                        original := latest.DeepCopy()
2✔
139
                        latest.Finalizers = slices.DeleteFunc(latest.Finalizers, func(s string) bool {
4✔
140
                                return s == targetFinalizer
2✔
141
                        })
2✔
142
                        if err := r.Patch(ctx, latest, client.MergeFrom(original)); err != nil {
2✔
143
                                return ctrl.Result{}, errLogAndWrap(log, err, "failed to remove finalizer from Target")
×
144
                        }
×
145
                }
146

147
                return ctrl.Result{}, nil
2✔
148
        }
149

150
        // Set finalizer if not set
151
        if !slices.Contains(target.Finalizers, targetFinalizer) {
213✔
152
                latest := &solarv1alpha1.Target{}
34✔
153
                if err := r.Get(ctx, req.NamespacedName, latest); err != nil {
34✔
154
                        return ctrl.Result{}, errLogAndWrap(log, err, "failed to get latest Target for finalizer addition")
×
155
                }
×
156

157
                original := latest.DeepCopy()
34✔
158
                latest.Finalizers = append(latest.Finalizers, targetFinalizer)
34✔
159
                if err := r.Patch(ctx, latest, client.MergeFrom(original)); err != nil {
34✔
160
                        return ctrl.Result{}, errLogAndWrap(log, err, "failed to add finalizer to Target")
×
161
                }
×
162

163
                return ctrl.Result{}, nil
34✔
164
        }
165

166
        // Resolve render registry — supports cross-namespace via ReferenceGrant
167
        registryNamespace := target.Namespace
145✔
168
        if target.Spec.RenderRegistryNamespace != "" {
145✔
169
                registryNamespace = target.Spec.RenderRegistryNamespace
×
170
        }
×
171

172
        // If the registry lives in a different namespace, verify a ReferenceGrant permits it
173
        // before attempting to fetch the object.
174
        if registryNamespace != target.Namespace {
145✔
175
                granted, err := r.registryGranted(ctx, registryNamespace, target.Namespace)
×
176
                if err != nil {
×
177
                        return ctrl.Result{}, errLogAndWrap(log, err, "failed to check ReferenceGrant for Registry")
×
178
                }
×
179
                if !granted {
×
180
                        if condErr := r.setCondition(ctx, target, ConditionTypeRegistryResolved, metav1.ConditionFalse, "NotGranted",
×
181
                                "No ReferenceGrant allows access to Registry "+target.Spec.RenderRegistryRef.Name+" in namespace "+registryNamespace); condErr != nil {
×
182
                                return ctrl.Result{}, condErr
×
183
                        }
×
184

185
                        return ctrl.Result{RequeueAfter: requeueAfterForCondition(
×
186
                                apimeta.FindStatusCondition(target.Status.Conditions, ConditionTypeRegistryResolved), time.Now())}, nil
×
187
                }
188
        }
189

190
        registry := &solarv1alpha1.Registry{}
145✔
191
        if err := r.Get(ctx, client.ObjectKey{
145✔
192
                Name:      target.Spec.RenderRegistryRef.Name,
145✔
193
                Namespace: registryNamespace,
145✔
194
        }, registry); err != nil {
165✔
195
                if apierrors.IsNotFound(err) {
40✔
196
                        if condErr := r.setCondition(ctx, target, ConditionTypeRegistryResolved, metav1.ConditionFalse, "NotFound",
20✔
197
                                "Registry not found: "+target.Spec.RenderRegistryRef.Name); condErr != nil {
20✔
198
                                return ctrl.Result{}, condErr
×
199
                        }
×
200

201
                        return ctrl.Result{RequeueAfter: requeueAfterForCondition(
20✔
202
                                apimeta.FindStatusCondition(target.Status.Conditions, ConditionTypeRegistryResolved), time.Now())}, nil
20✔
203
                }
204

205
                return ctrl.Result{}, errLogAndWrap(log, err, "failed to get Registry")
×
206
        }
207

208
        if registry.Spec.SolarSecretRef == nil {
127✔
209
                if condErr := r.setCondition(ctx, target, ConditionTypeRegistryResolved, metav1.ConditionFalse, "MissingSolarSecretRef",
2✔
210
                        "Registry does not have SolarSecretRef set, required for rendering"); condErr != nil {
2✔
211
                        return ctrl.Result{}, condErr
×
212
                }
×
213

214
                return ctrl.Result{}, nil
2✔
215
        }
216

217
        if condErr := r.setCondition(ctx, target, ConditionTypeRegistryResolved, metav1.ConditionTrue, "Resolved",
123✔
218
                "Registry resolved: "+registry.Name); condErr != nil {
123✔
219
                return ctrl.Result{}, condErr
×
220
        }
×
221

222
        // Build hostname→targetPullSecretName lookup from RegistryBindings for this target.
223
        pullSecretsByHost, err := r.buildPullSecretsLookup(ctx, target)
123✔
224
        if err != nil {
143✔
225
                if condErr := r.setCondition(ctx, target, ConditionTypeReleasesRendered, metav1.ConditionFalse, "RegistryBindingConflict",
20✔
226
                        err.Error()); condErr != nil {
20✔
227
                        return ctrl.Result{}, condErr
×
228
                }
×
229

230
                return ctrl.Result{}, errLogAndWrap(log, err, "failed to build pull secrets lookup from RegistryBindings")
20✔
231
        }
232

233
        // Collect ReleaseBindings for this target — same namespace first, then cross-namespace via ReferenceGrants.
234
        allBindings := &solarv1alpha1.ReleaseBindingList{}
103✔
235
        if err := r.APIReader.List(ctx, allBindings, client.InNamespace(target.Namespace)); err != nil {
103✔
236
                return ctrl.Result{}, errLogAndWrap(log, err, "failed to list ReleaseBindings")
×
237
        }
×
238
        bindingList := &solarv1alpha1.ReleaseBindingList{}
103✔
239
        for _, rb := range allBindings.Items {
216✔
240
                if rb.Spec.TargetRef.Name == target.Name && rb.Spec.TargetNamespace == "" {
226✔
241
                        bindingList.Items = append(bindingList.Items, rb)
113✔
242
                }
113✔
243
        }
244

245
        // Collect cross-namespace ReleaseBindings authorized by ReferenceGrants in target's namespace.
246
        crossNsBindings, crossNsErr := r.collectCrossNamespaceReleaseBindings(ctx, target)
103✔
247
        if crossNsErr != nil {
103✔
248
                return ctrl.Result{}, errLogAndWrap(log, crossNsErr, "failed to collect cross-namespace ReleaseBindings")
×
249
        }
×
250
        bindingList.Items = append(bindingList.Items, crossNsBindings...)
103✔
251

103✔
252
        // FIXME: collect cross-namespace RegistryBindings here once ADR-010 is finalized and
103✔
253
        // RegistryBinding collection is wired into the rendering pipeline.
103✔
254

103✔
255
        if len(bindingList.Items) == 0 {
111✔
256
                log.V(1).Info("No ReleaseBindings found for target")
8✔
257
                if condErr := r.setCondition(ctx, target, ConditionTypeReleasesRendered, metav1.ConditionFalse, "NoReleaseBindings",
8✔
258
                        "No ReleaseBindings found for this target"); condErr != nil {
8✔
259
                        return ctrl.Result{}, condErr
×
260
                }
×
261

262
                if condErr := r.setCondition(ctx, target, ConditionTypeReleasesResolved, metav1.ConditionFalse, "NoReleaseBindings",
8✔
263
                        "No ReleaseBindings found for this target"); condErr != nil {
8✔
264
                        return ctrl.Result{}, condErr
×
265
                }
×
266

267
                // Clean up any stale RenderTasks and RenderBindings left from prior reconciles.
268
                if err := r.deleteStaleRenderTasks(ctx, target, map[string]struct{}{}); err != nil {
8✔
269
                        return ctrl.Result{}, errLogAndWrap(log, err, "failed to clean up stale RenderTasks after all bindings removed")
×
270
                }
×
271
                if err := r.deleteStaleRenderBindings(ctx, target, map[string]struct{}{}); err != nil {
8✔
272
                        return ctrl.Result{}, errLogAndWrap(log, err, "failed to clean up stale RenderBindings after all bindings removed")
×
273
                }
×
274

275
                return ctrl.Result{}, nil
8✔
276
        }
277

278
        // For each bound release, ensure a per-release RenderTask exists
279
        var releases []releaseInfo
95✔
280

95✔
281
        pendingDeps := false
95✔
282

95✔
283
        for _, binding := range bindingList.Items {
215✔
284
                rel := &solarv1alpha1.Release{}
120✔
285
                if err := r.Get(ctx, client.ObjectKey{
120✔
286
                        Name:      binding.Spec.ReleaseRef.Name,
120✔
287
                        Namespace: binding.Namespace,
120✔
288
                }, rel); err != nil {
120✔
289
                        if apierrors.IsNotFound(err) {
×
290
                                log.V(1).Info("Release not found", "release", binding.Spec.ReleaseRef.Name)
×
291
                                pendingDeps = true
×
292

×
293
                                continue
×
294
                        }
295

296
                        return ctrl.Result{}, errLogAndWrap(log, err, "failed to get Release")
×
297
                }
298

299
                cv := &solarv1alpha1.ComponentVersion{}
120✔
300
                cvNamespace := rel.Namespace
120✔
301
                if rel.Spec.ComponentVersionNamespace != "" {
120✔
302
                        cvNamespace = rel.Spec.ComponentVersionNamespace
×
303
                }
×
304

305
                if cvNamespace != rel.Namespace {
120✔
306
                        granted := false
×
307
                        grantList := &solarv1alpha1.ReferenceGrantList{}
×
308
                        if err := r.List(ctx, grantList, client.InNamespace(cvNamespace)); err != nil {
×
309
                                return ctrl.Result{}, errLogAndWrap(log, err, "failed to check ReferenceGrant for cross-namespace ComponentVersion")
×
310
                        }
×
311
                        for i := range grantList.Items {
×
312
                                if grantPermitsComponentVersionAccess(&grantList.Items[i], rel.Namespace) {
×
313
                                        granted = true
×
314
                                        break
×
315
                                }
316
                        }
317
                        if !granted {
×
318
                                log.V(1).Info("ComponentVersion access not granted", "cv", rel.Spec.ComponentVersionRef.Name, "namespace", cvNamespace)
×
319
                                pendingDeps = true
×
320

×
321
                                continue
×
322
                        }
323
                }
324

325
                if err := r.Get(ctx, client.ObjectKey{
120✔
326
                        Name:      rel.Spec.ComponentVersionRef.Name,
120✔
327
                        Namespace: cvNamespace,
120✔
328
                }, cv); err != nil {
120✔
329
                        if apierrors.IsNotFound(err) {
×
330
                                log.V(1).Info("ComponentVersion not found", "cv", rel.Spec.ComponentVersionRef.Name)
×
331
                                pendingDeps = true
×
332

×
333
                                continue
×
334
                        }
335

336
                        return ctrl.Result{}, errLogAndWrap(log, err, "failed to get ComponentVersion")
×
337
                }
338

339
                rtName := releaseRenderTaskName(rel.Namespace, rel.Name, target.Name, rel.GetGeneration())
120✔
340
                releases = append(releases, releaseInfo{
120✔
341
                        bindingKey: binding.Namespace + "/" + binding.Name,
120✔
342
                        name:       rel.Name,
120✔
343
                        release:    rel,
120✔
344
                        cv:         cv,
120✔
345
                        rtName:     rtName,
120✔
346
                })
120✔
347
        }
348

349
        // Resolve conflicts: deduplicate by uniqueName (priority wins) and apply anti-affinity rules.
350
        var skipped []string
95✔
351
        releases, skipped = resolveReleaseConflicts(releases)
95✔
352
        if condErr := r.setResolvedCondition(ctx, target, skipped); condErr != nil {
95✔
353
                return ctrl.Result{}, condErr
×
354
        }
×
355

356
        if len(releases) == 0 && !pendingDeps {
95✔
357
                if condErr := r.setCondition(ctx, target, ConditionTypeReleasesRendered, metav1.ConditionFalse, "AllReleaseBindingsFiltered",
×
358
                        "All ReleaseBindings were filtered out by the release resolver (uniqueName conflicts or anti-affinity rules)"); condErr != nil {
×
359
                        return ctrl.Result{}, condErr
×
360
                }
×
361

362
                return ctrl.Result{}, nil
×
363
        }
364

365
        // Create per-release RenderTasks (one per target+release pair).
366
        // The renderer job handles dedup by skipping if the chart already exists in the registry.
367
        allRendered := true
95✔
368

95✔
369
        for i, ri := range releases {
206✔
370
                rt := &solarv1alpha1.RenderTask{}
111✔
371
                err := r.Get(ctx, client.ObjectKey{Name: ri.rtName, Namespace: target.Namespace}, rt)
111✔
372

111✔
373
                switch {
111✔
374
                case apierrors.IsNotFound(err):
29✔
375
                        spec, specErr := r.computeReleaseRenderTaskSpec(ri.release, ri.cv, registry, target, pullSecretsByHost)
29✔
376
                        if specErr != nil {
39✔
377
                                if condErr := r.setCondition(ctx, target, ConditionTypeReleasesRendered, metav1.ConditionFalse, "MissingRegistryBinding",
10✔
378
                                        specErr.Error()); condErr != nil {
10✔
379
                                        return ctrl.Result{}, condErr
×
380
                                }
×
381

382
                                return ctrl.Result{}, errLogAndWrap(log, specErr, "failed to compute release RenderTask spec")
10✔
383
                        }
384

385
                        rt = &solarv1alpha1.RenderTask{
19✔
386
                                ObjectMeta: metav1.ObjectMeta{
19✔
387
                                        Name:      ri.rtName,
19✔
388
                                        Namespace: target.Namespace,
19✔
389
                                },
19✔
390
                                Spec: spec,
19✔
391
                        }
19✔
392

19✔
393
                        if err := r.Create(ctx, rt); err != nil {
19✔
394
                                return ctrl.Result{}, errLogAndWrap(log, err, "failed to create release RenderTask")
×
395
                        }
×
396

397
                        log.V(1).Info("Created release RenderTask", "release", ri.name, "renderTask", ri.rtName)
19✔
398
                        r.Recorder.Eventf(target, nil, corev1.EventTypeNormal, "Created", "Create",
19✔
399
                                "Created release RenderTask %s for release %s", ri.rtName, ri.name)
19✔
400
                case err != nil:
×
401
                        return ctrl.Result{}, errLogAndWrap(log, err, "failed to get release RenderTask")
×
402
                default:
82✔
403
                        // RenderTask exists — check for spec drift (e.g. pull secrets
82✔
404
                        // changed after a RegistryBinding was created/updated).
82✔
405
                        desiredSpec, specErr := r.computeReleaseRenderTaskSpec(ri.release, ri.cv, registry, target, pullSecretsByHost)
82✔
406
                        if specErr != nil {
82✔
407
                                if condErr := r.setCondition(ctx, target, ConditionTypeReleasesRendered, metav1.ConditionFalse, "MissingRegistryBinding",
×
408
                                        specErr.Error()); condErr != nil {
×
409
                                        return ctrl.Result{}, condErr
×
410
                                }
×
411

412
                                return ctrl.Result{}, errLogAndWrap(log, specErr, "failed to compute release RenderTask spec for comparison")
×
413
                        }
414

415
                        if !apiequality.Semantic.DeepEqual(rt.Spec, desiredSpec) {
84✔
416
                                if err := r.Delete(ctx, rt); err != nil {
2✔
417
                                        return ctrl.Result{}, errLogAndWrap(log, err, "failed to delete stale release RenderTask")
×
418
                                }
×
419

420
                                rt = &solarv1alpha1.RenderTask{
2✔
421
                                        ObjectMeta: metav1.ObjectMeta{
2✔
422
                                                Name:      ri.rtName,
2✔
423
                                                Namespace: target.Namespace,
2✔
424
                                        },
2✔
425
                                        Spec: desiredSpec,
2✔
426
                                }
2✔
427

2✔
428
                                if err := r.Create(ctx, rt); err != nil {
2✔
429
                                        return ctrl.Result{}, errLogAndWrap(log, err, "failed to recreate release RenderTask")
×
430
                                }
×
431

432
                                log.V(1).Info("Recreated release RenderTask (spec drift)", "release", ri.name, "renderTask", ri.rtName)
2✔
433
                                r.Recorder.Eventf(target, nil, corev1.EventTypeNormal, "Updated", "Update",
2✔
434
                                        "Recreated release RenderTask %s for release %s (spec drift)", ri.rtName, ri.name)
2✔
435
                        }
436
                }
437

438
                // Check if release RenderTask is complete
439
                if apimeta.IsStatusConditionTrue(rt.Status.Conditions, ConditionTypeJobFailed) {
101✔
440
                        if condErr := r.setCondition(ctx, target, ConditionTypeReleasesRendered, metav1.ConditionFalse, "ReleaseFailed",
×
441
                                fmt.Sprintf("Release %s rendering failed", ri.name)); condErr != nil {
×
442
                                return ctrl.Result{}, condErr
×
443
                        }
×
444

445
                        return ctrl.Result{}, nil
×
446
                }
447

448
                if apimeta.IsStatusConditionTrue(rt.Status.Conditions, ConditionTypeJobSucceeded) && rt.Status.ChartURL != "" {
143✔
449
                        releases[i].chartURL = rt.Status.ChartURL
42✔
450

42✔
451
                        // Ensure a RenderArtifact object exists for the pushed OCI artifact, and
42✔
452
                        // create a RenderBinding linking this Target to it.
42✔
453
                        aName := renderArtifactName(target.Namespace, rt.Spec.BaseURL, rt.Spec.Repository, rt.Spec.Tag)
42✔
454
                        bName := renderBindingName(aName, target.Name)
42✔
455
                        // Create the RenderBinding before the RenderArtifact to avoid a race
42✔
456
                        if err := r.ensureRenderBinding(ctx, target, aName, bName); err != nil {
42✔
457
                                return ctrl.Result{}, errLogAndWrap(log, err, "failed to ensure RenderBinding for release")
×
458
                        }
×
459
                        if err := r.ensureRenderArtifact(ctx, aName, rt, registry.Spec.Flavor, registryNamespace); err != nil {
42✔
460
                                return ctrl.Result{}, errLogAndWrap(log, err, "failed to ensure RenderArtifact for release")
×
461
                        }
×
462
                        releases[i].artifactName = aName
42✔
463
                        releases[i].artifactBindingName = bName
42✔
464
                } else {
59✔
465
                        allRendered = false
59✔
466
                }
59✔
467
        }
468

469
        if pendingDeps {
85✔
470
                if condErr := r.setCondition(ctx, target, ConditionTypeReleasesRendered, metav1.ConditionFalse, "MissingDependencies",
×
471
                        "One or more bound Releases or ComponentVersions not found"); condErr != nil {
×
472
                        return ctrl.Result{}, condErr
×
473
                }
×
474

475
                return ctrl.Result{RequeueAfter: requeueAfterForCondition(
×
476
                        apimeta.FindStatusCondition(target.Status.Conditions, ConditionTypeReleasesRendered), time.Now())}, nil
×
477
        }
478

479
        if !allRendered {
141✔
480
                if condErr := r.setCondition(ctx, target, ConditionTypeReleasesRendered, metav1.ConditionFalse, "Pending",
56✔
481
                        "Waiting for release RenderTasks to complete"); condErr != nil {
58✔
482
                        return ctrl.Result{}, condErr
2✔
483
                }
2✔
484

485
                return ctrl.Result{RequeueAfter: requeueAfterForCondition(
54✔
486
                        apimeta.FindStatusCondition(target.Status.Conditions, ConditionTypeReleasesRendered), time.Now())}, nil
54✔
487
        }
488

489
        if condErr := r.setCondition(ctx, target, ConditionTypeReleasesRendered, metav1.ConditionTrue, "AllRendered",
29✔
490
                "All releases rendered successfully"); condErr != nil {
29✔
491
                return ctrl.Result{}, condErr
×
492
        }
×
493

494
        // Determine if a new bootstrap render is needed by checking whether the
495
        // current bootstrapVersion's RenderTask still matches the desired release set.
496
        bootstrapVersion := target.Status.BootstrapVersion
29✔
497
        bootstrapRTName := targetRenderTaskName(target.Name, bootstrapVersion)
29✔
498
        bootstrapRT := &solarv1alpha1.RenderTask{}
29✔
499
        err = r.Get(ctx, client.ObjectKey{Name: bootstrapRTName, Namespace: target.Namespace}, bootstrapRT)
29✔
500

29✔
501
        needsNewBootstrap := false
29✔
502

29✔
503
        switch {
29✔
504
        case apierrors.IsNotFound(err):
5✔
505
                // No RenderTask for the current version yet — create one
5✔
506
                needsNewBootstrap = true
5✔
507
        case err != nil:
×
508
                return ctrl.Result{}, errLogAndWrap(log, err, "failed to get bootstrap RenderTask")
×
509
        default:
24✔
510
                // RenderTask exists — check if the desired bootstrap input changed
24✔
511
                // (release set, resolved refs/tags, or userdata)
24✔
512
                desiredInput, inputErr := buildBootstrapInput(target, releases, registry.Spec.TargetPullSecretName)
24✔
513
                if inputErr != nil {
24✔
514
                        return ctrl.Result{}, errLogAndWrap(log, inputErr, "failed to build desired bootstrap input for comparison")
×
515
                }
×
516

517
                existingInput := bootstrapRT.Spec.RendererConfig.BootstrapConfig.Input
24✔
518
                if !apiequality.Semantic.DeepEqual(desiredInput, existingInput) {
26✔
519
                        bootstrapVersion++
2✔
520
                        needsNewBootstrap = true
2✔
521
                }
2✔
522
        }
523

524
        if needsNewBootstrap {
36✔
525
                spec, specErr := r.computeBootstrapRenderTaskSpec(target, releases, registry, bootstrapVersion)
7✔
526
                if specErr != nil {
7✔
527
                        return ctrl.Result{}, errLogAndWrap(log, specErr, "failed to compute bootstrap RenderTask spec")
×
528
                }
×
529

530
                bootstrapRTName = targetRenderTaskName(target.Name, bootstrapVersion)
7✔
531
                bootstrapRT = &solarv1alpha1.RenderTask{
7✔
532
                        ObjectMeta: metav1.ObjectMeta{
7✔
533
                                Name:      bootstrapRTName,
7✔
534
                                Namespace: target.Namespace,
7✔
535
                        },
7✔
536
                        Spec: spec,
7✔
537
                }
7✔
538

7✔
539
                if err := r.Create(ctx, bootstrapRT); err != nil {
7✔
540
                        if !apierrors.IsAlreadyExists(err) {
×
541
                                return ctrl.Result{}, errLogAndWrap(log, err, "failed to create bootstrap RenderTask")
×
542
                        }
×
543

544
                        if err := r.Get(ctx, client.ObjectKey{Name: bootstrapRTName, Namespace: target.Namespace}, bootstrapRT); err != nil {
×
545
                                return ctrl.Result{}, errLogAndWrap(log, err, "failed to get existing bootstrap RenderTask")
×
546
                        }
×
547
                } else {
7✔
548
                        log.V(1).Info("Created bootstrap RenderTask", "renderTask", bootstrapRTName, "bootstrapVersion", bootstrapVersion)
7✔
549
                        r.Recorder.Eventf(target, nil, corev1.EventTypeNormal, "Created", "Create",
7✔
550
                                "Created bootstrap RenderTask %s (version %d)", bootstrapRTName, bootstrapVersion)
7✔
551
                }
7✔
552

553
                // Persist the new bootstrapVersion in status
554
                if bootstrapVersion != target.Status.BootstrapVersion {
9✔
555
                        target.Status.BootstrapVersion = bootstrapVersion
2✔
556
                        if err := r.Status().Update(ctx, target); err != nil {
2✔
557
                                return ctrl.Result{}, errLogAndWrap(log, err, "failed to update Target bootstrapVersion")
×
558
                        }
×
559
                }
560
        }
561

562
        // Update target status from bootstrap RenderTask
563
        if apimeta.IsStatusConditionTrue(bootstrapRT.Status.Conditions, ConditionTypeJobFailed) {
29✔
564
                if condErr := r.setCondition(ctx, target, ConditionTypeBootstrapReady, metav1.ConditionFalse, "Failed",
×
565
                        "Bootstrap rendering failed"); condErr != nil {
×
566
                        return ctrl.Result{}, condErr
×
567
                }
×
568

569
                return ctrl.Result{}, nil
×
570
        }
571

572
        if apimeta.IsStatusConditionTrue(bootstrapRT.Status.Conditions, ConditionTypeJobSucceeded) {
37✔
573
                if condErr := r.setCondition(ctx, target, ConditionTypeBootstrapReady, metav1.ConditionTrue, "Ready",
8✔
574
                        "Bootstrap rendered successfully: "+bootstrapRT.Status.ChartURL); condErr != nil {
8✔
575
                        return ctrl.Result{}, condErr
×
576
                }
×
577

578
                // Ensure RenderArtifact + RenderBinding exist for the bootstrap chart.
579
                bootstrapArtifactName := renderArtifactName(target.Namespace, bootstrapRT.Spec.BaseURL, bootstrapRT.Spec.Repository, bootstrapRT.Spec.Tag)
8✔
580
                bootstrapBindingName := renderBindingName(bootstrapArtifactName, target.Name)
8✔
581
                // Create the RenderBinding before the RenderArtifact to avoid a race
8✔
582
                if err := r.ensureRenderBinding(ctx, target, bootstrapArtifactName, bootstrapBindingName); err != nil {
8✔
583
                        return ctrl.Result{}, errLogAndWrap(log, err, "failed to ensure RenderBinding for bootstrap")
×
584
                }
×
585
                if err := r.ensureRenderArtifact(ctx, bootstrapArtifactName, bootstrapRT, registry.Spec.Flavor, registryNamespace); err != nil {
8✔
586
                        return ctrl.Result{}, errLogAndWrap(log, err, "failed to ensure RenderArtifact for bootstrap")
×
587
                }
×
588

589
                // Clean up stale RenderTasks owned by this target (old versions)
590
                currentRTNames := map[string]struct{}{bootstrapRTName: {}}
8✔
591
                for _, ri := range releases {
20✔
592
                        currentRTNames[ri.rtName] = struct{}{}
12✔
593
                }
12✔
594
                if err := r.deleteStaleRenderTasks(ctx, target, currentRTNames); err != nil {
8✔
595
                        // Stale cleanup is best-effort: a failure here does not affect the desired state
×
596
                        // that was just reconciled. The next reconcile will retry the cleanup.
×
597
                        log.Error(err, "failed to clean up stale RenderTasks")
×
598
                }
×
599

600
                // Clean up stale RenderBindings owned by this target.
601
                currentBindingNames := map[string]struct{}{bootstrapBindingName: {}}
8✔
602
                for _, ri := range releases {
20✔
603
                        if ri.artifactBindingName != "" {
24✔
604
                                currentBindingNames[ri.artifactBindingName] = struct{}{}
12✔
605
                        }
12✔
606
                }
607
                if err := r.deleteStaleRenderBindings(ctx, target, currentBindingNames); err != nil {
8✔
608
                        // Stale cleanup is best-effort: a failure here does not affect the desired state
×
609
                        // that was just reconciled. The next reconcile will retry the cleanup.
×
610
                        log.Error(err, "failed to clean up stale RenderBindings")
×
611
                }
×
612

613
                return ctrl.Result{RequeueAfter: 30 * time.Second}, nil
8✔
614
        }
615

616
        // Still running
617
        return ctrl.Result{RequeueAfter: 30 * time.Second}, nil
21✔
618
}
619

620
func (r *TargetReconciler) setCondition(ctx context.Context, target *solarv1alpha1.Target, condType string, status metav1.ConditionStatus, reason, message string) error {
379✔
621
        changed := apimeta.SetStatusCondition(&target.Status.Conditions, metav1.Condition{
379✔
622
                Type:               condType,
379✔
623
                Status:             status,
379✔
624
                ObservedGeneration: target.Generation,
379✔
625
                Reason:             reason,
379✔
626
                Message:            message,
379✔
627
        })
379✔
628
        if changed {
472✔
629
                if err := r.Status().Update(ctx, target); err != nil {
95✔
630
                        return fmt.Errorf("failed to update Target status condition %s: %w", condType, err)
2✔
631
                }
2✔
632
        }
633

634
        return nil
377✔
635
}
636

637
func (r *TargetReconciler) setResolvedCondition(ctx context.Context, target *solarv1alpha1.Target, skipped []string) error {
95✔
638
        if len(skipped) == 0 {
181✔
639
                return r.setCondition(ctx, target, ConditionTypeReleasesResolved, metav1.ConditionTrue, "NoConflicts", "")
86✔
640
        }
86✔
641

642
        return r.setCondition(ctx, target, ConditionTypeReleasesResolved, metav1.ConditionTrue, "Resolved", strings.Join(skipped, "; "))
9✔
643
}
644

645
// resolveReleaseConflicts deduplicates releases by uniqueName (keeping the highest-priority
646
// binding) and filters releases that violate anti-affinity rules of already-accepted releases.
647
// Releases without a uniqueName are deduplicated using the parent Component name from the CV.
648
// It returns the accepted releases and a slice of human-readable filter messages.
649
func resolveReleaseConflicts(releases []releaseInfo) ([]releaseInfo, []string) {
104✔
650
        if len(releases) == 0 {
105✔
651
                return releases, nil
1✔
652
        }
1✔
653

654
        // Step A: uniqueName deduplication.
655
        // When UniqueName is empty, fall back to the parent Component name from the CV.
656
        namedGroups := map[string][]releaseInfo{}
103✔
657

103✔
658
        for i, ri := range releases {
237✔
659
                uname := effectiveUniqueName(ri.release, ri.cv)
134✔
660
                releases[i].uniqueName = uname
134✔
661
                namedGroups[uname] = append(namedGroups[uname], releases[i])
134✔
662
        }
134✔
663

664
        var accepted []releaseInfo
103✔
665

103✔
666
        var skipped []string
103✔
667

103✔
668
        // byPriority sorts releases with highest priority first; bindingKey breaks ties.
103✔
669
        byPriority := func(a, b releaseInfo) bool {
134✔
670
                if a.release.Spec.Priority != b.release.Spec.Priority {
41✔
671
                        return a.release.Spec.Priority > b.release.Spec.Priority
10✔
672
                }
10✔
673

674
                return a.bindingKey < b.bindingKey
21✔
675
        }
676

677
        uniqueNames := make([]string, 0, len(namedGroups))
103✔
678
        for k := range namedGroups {
228✔
679
                uniqueNames = append(uniqueNames, k)
125✔
680
        }
125✔
681

682
        sort.Strings(uniqueNames)
103✔
683

103✔
684
        for _, uniqueName := range uniqueNames {
228✔
685
                group := namedGroups[uniqueName]
125✔
686
                sort.Slice(group, func(i, j int) bool { return byPriority(group[i], group[j]) })
134✔
687

688
                accepted = append(accepted, group[0])
125✔
689

125✔
690
                for _, loser := range group[1:] {
134✔
691
                        skipped = append(skipped, fmt.Sprintf(
9✔
692
                                "binding %s filtered: uniqueName %q conflict, lower priority than %s",
9✔
693
                                loser.bindingKey, uniqueName, group[0].bindingKey,
9✔
694
                        ))
9✔
695
                }
9✔
696
        }
697

698
        // Step B: anti-affinity evaluation.
699
        // Walk in deterministic order (priority desc, bindingKey asc); accept each release only
700
        // if its AntiAffinity selector does not match any already-accepted release's labels.
701
        sort.Slice(accepted, func(i, j int) bool { return byPriority(accepted[i], accepted[j]) })
125✔
702

703
        resolved := make([]releaseInfo, 0, len(accepted))
103✔
704

103✔
705
        for _, ri := range accepted {
228✔
706
                // Parse ri's own anti-affinity selector once; bail early on invalid selector.
125✔
707
                var riSelector labels.Selector
125✔
708
                if ri.release.Spec.AntiAffinity != nil {
132✔
709
                        sel, err := metav1.LabelSelectorAsSelector(ri.release.Spec.AntiAffinity)
7✔
710
                        if err != nil {
8✔
711
                                skipped = append(skipped, fmt.Sprintf(
1✔
712
                                        "binding %s filtered: invalid antiAffinity selector: %v",
1✔
713
                                        ri.bindingKey, err,
1✔
714
                                ))
1✔
715

1✔
716
                                continue
1✔
717
                        }
718

719
                        riSelector = sel
6✔
720
                }
721

722
                // Check both directions: ri's anti-affinity against already-resolved labels,
723
                // and already-resolved anti-affinities against ri's labels.
724
                conflict := ""
124✔
725
                for _, other := range resolved {
146✔
726
                        if riSelector != nil && riSelector.Matches(labels.Set(other.release.Labels)) {
26✔
727
                                conflict = other.bindingKey
4✔
728
                                break
4✔
729
                        }
730

731
                        if other.release.Spec.AntiAffinity != nil {
19✔
732
                                otherSel, err := metav1.LabelSelectorAsSelector(other.release.Spec.AntiAffinity)
1✔
733
                                if err == nil && otherSel.Matches(labels.Set(ri.release.Labels)) {
2✔
734
                                        conflict = other.bindingKey
1✔
735
                                        break
1✔
736
                                }
737
                        }
738
                }
739

740
                if conflict != "" {
129✔
741
                        skipped = append(skipped, fmt.Sprintf(
5✔
742
                                "binding %s filtered: anti-affinity conflict with %s",
5✔
743
                                ri.bindingKey, conflict,
5✔
744
                        ))
5✔
745
                } else {
124✔
746
                        resolved = append(resolved, ri)
119✔
747
                }
119✔
748
        }
749

750
        return resolved, skipped
103✔
751
}
752

753
// deleteStaleRenderTasks removes RenderTasks owned by this target that are no
754
// longer needed. Any owned RenderTask whose name is not in currentRTNames is
755
// deleted. This covers both old bootstrap versions and old release generations.
756
func (r *TargetReconciler) deleteStaleRenderTasks(ctx context.Context, target *solarv1alpha1.Target, currentRTNames map[string]struct{}) error {
16✔
757
        log := ctrl.LoggerFrom(ctx)
16✔
758

16✔
759
        rtList := &solarv1alpha1.RenderTaskList{}
16✔
760
        if err := r.List(ctx, rtList,
16✔
761
                client.InNamespace(target.Namespace),
16✔
762
                client.MatchingFields{indexOwnerKind: "Target"},
16✔
763
        ); err != nil {
16✔
764
                return err
×
765
        }
×
766

767
        for i := range rtList.Items {
39✔
768
                rt := &rtList.Items[i]
23✔
769
                if rt.Spec.OwnerName != target.Name || rt.Spec.OwnerNamespace != target.Namespace {
23✔
770
                        continue
×
771
                }
772

773
                if _, current := currentRTNames[rt.Name]; current {
43✔
774
                        continue
20✔
775
                }
776

777
                log.V(1).Info("Deleting stale RenderTask", "renderTask", rt.Name)
3✔
778
                if err := r.Delete(ctx, rt, client.PropagationPolicy(metav1.DeletePropagationBackground)); client.IgnoreNotFound(err) != nil {
3✔
779
                        return err
×
780
                }
×
781

782
                r.Recorder.Eventf(target, nil, corev1.EventTypeNormal, "Deleted", "Delete",
3✔
783
                        "Deleted stale RenderTask %s", rt.Name)
3✔
784
        }
785

786
        return nil
16✔
787
}
788

789
func (r *TargetReconciler) deleteOwnedRenderTasks(ctx context.Context, target *solarv1alpha1.Target) error {
2✔
790
        rtList := &solarv1alpha1.RenderTaskList{}
2✔
791
        if err := r.List(ctx, rtList,
2✔
792
                client.InNamespace(target.Namespace),
2✔
793
                client.MatchingFields{indexOwnerKind: "Target"},
2✔
794
        ); err != nil {
2✔
795
                return err
×
796
        }
×
797

798
        for i := range rtList.Items {
4✔
799
                rt := &rtList.Items[i]
2✔
800
                if rt.Spec.OwnerName == target.Name && rt.Spec.OwnerNamespace == target.Namespace {
4✔
801
                        if err := r.Delete(ctx, rt, client.PropagationPolicy(metav1.DeletePropagationBackground)); client.IgnoreNotFound(err) != nil {
2✔
802
                                return err
×
803
                        }
×
804
                }
805
        }
806

807
        return nil
2✔
808
}
809

810
// deleteStaleRenderBindings removes RenderBindings owned by this target that are no
811
// longer needed (artifact is not in currentBindingNames).
812
func (r *TargetReconciler) deleteStaleRenderBindings(ctx context.Context, target *solarv1alpha1.Target, currentBindingNames map[string]struct{}) error {
16✔
813
        log := ctrl.LoggerFrom(ctx)
16✔
814

16✔
815
        bindingList := &solarv1alpha1.RenderBindingList{}
16✔
816
        if err := r.APIReader.List(ctx, bindingList, client.InNamespace(target.Namespace)); err != nil {
16✔
817
                return err
×
818
        }
×
819

820
        for i := range bindingList.Items {
39✔
821
                b := &bindingList.Items[i]
23✔
822
                if b.Spec.OwnerKind != "Target" || b.Spec.OwnerName != target.Name || b.Spec.OwnerNamespace != target.Namespace {
23✔
823
                        continue
×
824
                }
825

826
                if _, current := currentBindingNames[b.Name]; current {
43✔
827
                        continue
20✔
828
                }
829

830
                log.V(1).Info("Deleting stale RenderBinding", "renderBinding", b.Name)
3✔
831
                if err := r.Delete(ctx, b); client.IgnoreNotFound(err) != nil {
3✔
832
                        return err
×
833
                }
×
834
        }
835

836
        return nil
16✔
837
}
838

839
// deleteOwnedRenderBindings removes all RenderBindings owned by this target.
840
// Called during Target deletion to trigger GC of any associated RenderArtifacts.
841
func (r *TargetReconciler) deleteOwnedRenderBindings(ctx context.Context, target *solarv1alpha1.Target) error {
2✔
842
        bindingList := &solarv1alpha1.RenderBindingList{}
2✔
843
        if err := r.APIReader.List(ctx, bindingList, client.InNamespace(target.Namespace)); err != nil {
2✔
844
                return err
×
845
        }
×
846

847
        for i := range bindingList.Items {
3✔
848
                b := &bindingList.Items[i]
1✔
849
                if b.Spec.OwnerKind == "Target" && b.Spec.OwnerName == target.Name && b.Spec.OwnerNamespace == target.Namespace {
2✔
850
                        if err := r.Delete(ctx, b); client.IgnoreNotFound(err) != nil {
1✔
851
                                return err
×
852
                        }
×
853
                }
854
        }
855

856
        return nil
2✔
857
}
858

859
// ensureRenderArtifact creates a RenderArtifact for the given RenderTask's OCI coordinates
860
// if one does not already exist. If it already exists (possibly created by another Target
861
// reconciling the same shared artifact), it refreshes the push-secret coordinates to this
862
// target's known-good credentials — see refreshRenderArtifactSecret.
863
//
864
// pushSecretNamespace is passed explicitly because the secret may live in a different
865
// namespace than the RenderTask (e.g. a cluster-scoped secret namespace chosen by the
866
// operator). It must not be inferred from rt.Namespace.
867
func (r *TargetReconciler) ensureRenderArtifact(ctx context.Context, name string, rt *solarv1alpha1.RenderTask, flavor, pushSecretNamespace string) error {
52✔
868
        artifact := &solarv1alpha1.RenderArtifact{}
52✔
869
        if err := r.Get(ctx, client.ObjectKey{Name: name, Namespace: rt.Namespace}, artifact); err == nil {
93✔
870
                if !artifact.DeletionTimestamp.IsZero() {
41✔
871
                        // The artifact is terminating (OCI cleanup in progress). Creating a binding
×
872
                        // against it would race with the finalizer. Requeue and wait for full deletion.
×
873
                        return fmt.Errorf("RenderArtifact %s/%s is terminating; requeuing", rt.Namespace, name)
×
874
                }
×
875

876
                return r.refreshRenderArtifactSecret(ctx, artifact, rt.Spec.PushSecretRef, pushSecretNamespace, flavor)
41✔
877
        } else if !apierrors.IsNotFound(err) {
11✔
878
                return err
×
879
        }
×
880

881
        artifact = &solarv1alpha1.RenderArtifact{
11✔
882
                ObjectMeta: metav1.ObjectMeta{
11✔
883
                        Name:      name,
11✔
884
                        Namespace: rt.Namespace,
11✔
885
                },
11✔
886
                Spec: solarv1alpha1.RenderArtifactSpec{
11✔
887
                        BaseURL:             rt.Spec.BaseURL,
11✔
888
                        Repository:          rt.Spec.Repository,
11✔
889
                        Tag:                 rt.Spec.Tag,
11✔
890
                        RenderTaskRef:       rt.Name,
11✔
891
                        PushSecretRef:       rt.Spec.PushSecretRef,
11✔
892
                        PushSecretNamespace: pushSecretNamespace,
11✔
893
                        RegistryFlavor:      flavor,
11✔
894
                },
11✔
895
        }
11✔
896

11✔
897
        if err := r.Create(ctx, artifact); err != nil && !apierrors.IsAlreadyExists(err) {
11✔
898
                return err
×
899
        }
×
900

901
        return nil
11✔
902
}
903

904
// refreshRenderArtifactSecret heals a shared RenderArtifact's push-secret coordinates when they
905
// no longer resolve. A RenderArtifact is shared by every Target that renders the same chart to
906
// the same registry, but its PushSecretRef/PushSecretNamespace are only the values of whichever
907
// Target created it. If that creator used a namespace that is later torn down, the reference
908
// dangles and OCI cleanup can never authenticate to delete the tag (it would leak, or the object
909
// wedges in Terminating).
910
//
911
// We therefore re-pin the coordinates to this Target's known-good credentials — the Target just
912
// rendered/pushed with them — but ONLY when the artifact's current reference does not resolve.
913
// Healing only on breakage is important: overwriting on every reconcile would let a Target whose
914
// secret lives in an ephemeral namespace hijack the reference away from a perfectly valid one.
915
// As long as any referencing Target has working credentials, the artifact self-heals to them
916
// before it is ever garbage-collected.
917
func (r *TargetReconciler) refreshRenderArtifactSecret(ctx context.Context, artifact *solarv1alpha1.RenderArtifact, pushSecretRef *corev1.LocalObjectReference, pushSecretNamespace, flavor string) error {
41✔
918
        needsSecretRepin := !r.renderArtifactSecretResolves(ctx, artifact)
41✔
919
        flavorDrift := artifact.Spec.RegistryFlavor != flavor
41✔
920

41✔
921
        if !needsSecretRepin && !flavorDrift {
42✔
922
                return nil
1✔
923
        }
1✔
924

925
        base := artifact.DeepCopy()
40✔
926
        if needsSecretRepin {
80✔
927
                artifact.Spec.PushSecretRef = pushSecretRef
40✔
928
                artifact.Spec.PushSecretNamespace = pushSecretNamespace
40✔
929
        }
40✔
930
        artifact.Spec.RegistryFlavor = flavor
40✔
931
        if err := r.Patch(ctx, artifact, client.MergeFrom(base)); err != nil {
40✔
NEW
932
                return fmt.Errorf("failed to refresh push-secret coordinates on RenderArtifact %s/%s: %w",
×
NEW
933
                        artifact.Namespace, artifact.Name, err)
×
NEW
934
        }
×
935

936
        if needsSecretRepin {
80✔
937
                ctrl.LoggerFrom(ctx).V(1).Info("Healed dangling push-secret coordinates on shared RenderArtifact",
40✔
938
                        "artifact", artifact.Name, "pushSecretNamespace", pushSecretNamespace)
40✔
939
        }
40✔
940

941
        return nil
40✔
942
}
943

944
// renderArtifactSecretResolves reports whether the artifact's currently-referenced push secret
945
// can be read. A nil/empty PushSecretRef means anonymous push, which always "resolves".
946
func (r *TargetReconciler) renderArtifactSecretResolves(ctx context.Context, artifact *solarv1alpha1.RenderArtifact) bool {
41✔
947
        if artifact.Spec.PushSecretRef == nil || artifact.Spec.PushSecretRef.Name == "" {
41✔
NEW
948
                return true
×
NEW
949
        }
×
950

951
        secretNs := artifact.Spec.PushSecretNamespace
41✔
952
        if secretNs == "" {
41✔
NEW
953
                secretNs = artifact.Namespace
×
NEW
954
        }
×
955

956
        err := r.Get(ctx, client.ObjectKey{Name: artifact.Spec.PushSecretRef.Name, Namespace: secretNs}, &corev1.Secret{})
41✔
957

41✔
958
        return err == nil
41✔
959
}
960

961
// ensureRenderBinding creates a RenderBinding linking this Target to the named
962
// RenderArtifact if one does not already exist. Idempotent.
963
func (r *TargetReconciler) ensureRenderBinding(ctx context.Context, target *solarv1alpha1.Target, artifactName, bindingName string) error {
50✔
964
        binding := &solarv1alpha1.RenderBinding{}
50✔
965
        if err := r.Get(ctx, client.ObjectKey{Name: bindingName, Namespace: target.Namespace}, binding); err == nil {
89✔
966
                return nil
39✔
967
        } else if !apierrors.IsNotFound(err) {
50✔
968
                return err
×
969
        }
×
970

971
        binding = &solarv1alpha1.RenderBinding{
11✔
972
                ObjectMeta: metav1.ObjectMeta{
11✔
973
                        Name:      bindingName,
11✔
974
                        Namespace: target.Namespace,
11✔
975
                },
11✔
976
                Spec: solarv1alpha1.RenderBindingSpec{
11✔
977
                        RenderArtifactRef: corev1.LocalObjectReference{Name: artifactName},
11✔
978
                        OwnerKind:         "Target",
11✔
979
                        OwnerName:         target.Name,
11✔
980
                        OwnerNamespace:    target.Namespace,
11✔
981
                },
11✔
982
        }
11✔
983

11✔
984
        if err := r.Create(ctx, binding); err != nil && !apierrors.IsAlreadyExists(err) {
11✔
985
                return err
×
986
        }
×
987

988
        return nil
11✔
989
}
990

991
func (r *TargetReconciler) computeReleaseRenderTaskSpec(rel *solarv1alpha1.Release, cv *solarv1alpha1.ComponentVersion, registry *solarv1alpha1.Registry, target *solarv1alpha1.Target, pullSecretsByHost map[string]string) (solarv1alpha1.RenderTaskSpec, error) {
111✔
992
        chartName := fmt.Sprintf("release-%s", rel.Name)
111✔
993
        repo := fmt.Sprintf("%s/%s/%s", target.Namespace, rel.Namespace, chartName)
111✔
994

111✔
995
        var targetNamespace string
111✔
996
        if rel.Spec.TargetNamespace != nil {
222✔
997
                targetNamespace = *rel.Spec.TargetNamespace
111✔
998
        }
111✔
999

1000
        resolvedResources, err := resolveResources(cv.Spec.Resources, pullSecretsByHost, r.RegistryBindingStrict)
111✔
1001
        if err != nil {
121✔
1002
                return solarv1alpha1.RenderTaskSpec{}, fmt.Errorf("release %s: %w", rel.Name, err)
10✔
1003
        }
10✔
1004

1005
        // Include a hash of pull-secret names in the tag so that charts whose
1006
        // content differs only in secretRef get unique OCI tags. Without this,
1007
        // the renderer's exists-check skips re-pushing after a spec-drift
1008
        // recreation (e.g. RegistryBinding created after the first render).
1009
        tag := fmt.Sprintf("v0.0.%d-%s", rel.GetGeneration(), pullSecretsTag(resolvedResources))
101✔
1010

101✔
1011
        return solarv1alpha1.RenderTaskSpec{
101✔
1012
                RendererConfig: solarv1alpha1.RendererConfig{
101✔
1013
                        Type: solarv1alpha1.RendererConfigTypeRelease,
101✔
1014
                        ReleaseConfig: solarv1alpha1.ReleaseConfig{
101✔
1015
                                Chart: solarv1alpha1.ChartConfig{
101✔
1016
                                        Name:        chartName,
101✔
1017
                                        Description: fmt.Sprintf("Release of %s", rel.Spec.ComponentVersionRef.Name),
101✔
1018
                                        Version:     tag,
101✔
1019
                                        AppVersion:  tag,
101✔
1020
                                },
101✔
1021
                                Input: solarv1alpha1.ReleaseInput{
101✔
1022
                                        Component:  solarv1alpha1.ReleaseComponent{Name: cv.Spec.ComponentRef.Name},
101✔
1023
                                        Resources:  resolvedResources,
101✔
1024
                                        Entrypoint: cv.Spec.Entrypoint,
101✔
1025
                                },
101✔
1026
                                Values:          rel.Spec.Values,
101✔
1027
                                TargetNamespace: targetNamespace,
101✔
1028
                        },
101✔
1029
                },
101✔
1030
                Repository:     repo,
101✔
1031
                Tag:            tag,
101✔
1032
                BaseURL:        registry.Spec.Hostname,
101✔
1033
                PushSecretRef:  registry.Spec.SolarSecretRef,
101✔
1034
                FailedJobTTL:   rel.Spec.FailedJobTTL,
101✔
1035
                OwnerName:      target.Name,
101✔
1036
                OwnerNamespace: target.Namespace,
101✔
1037
                OwnerKind:      "Target",
101✔
1038
        }, nil
101✔
1039
}
1040

1041
// buildBootstrapInput constructs the desired BootstrapInput from the current
1042
// target and resolved releases. Used for both comparison and spec construction.
1043
func buildBootstrapInput(target *solarv1alpha1.Target, releases []releaseInfo, renderRegistryPullSecret string) (solarv1alpha1.BootstrapInput, error) {
33✔
1044
        resolvedReleases := map[string]solarv1alpha1.ResolvedResourceAccess{}
33✔
1045

33✔
1046
        for _, ri := range releases {
78✔
1047
                if ri.uniqueName == "" {
46✔
1048
                        return solarv1alpha1.BootstrapInput{}, fmt.Errorf("release %q has empty uniqueName; resolveReleaseConflicts must run before buildBootstrapInput", ri.name)
1✔
1049
                }
1✔
1050

1051
                ref, err := ociname.ParseReference(ri.chartURL)
44✔
1052
                if err != nil {
44✔
1053
                        return solarv1alpha1.BootstrapInput{}, fmt.Errorf("failed to parse chartURL %s: %w", ri.chartURL, err)
×
1054
                }
×
1055

1056
                repo, err := url.JoinPath(ref.Context().RegistryStr(), ref.Context().RepositoryStr())
44✔
1057
                if err != nil {
44✔
1058
                        return solarv1alpha1.BootstrapInput{}, err
×
1059
                }
×
1060

1061
                resolvedReleases[ri.uniqueName] = solarv1alpha1.ResolvedResourceAccess{
44✔
1062
                        Repository:     strings.TrimPrefix(repo, "oci://"),
44✔
1063
                        Tag:            ref.Identifier(),
44✔
1064
                        PullSecretName: renderRegistryPullSecret,
44✔
1065
                }
44✔
1066
        }
1067

1068
        return solarv1alpha1.BootstrapInput{
32✔
1069
                Releases: resolvedReleases,
32✔
1070
                Userdata: target.Spec.Userdata,
32✔
1071
        }, nil
32✔
1072
}
1073

1074
func (r *TargetReconciler) computeBootstrapRenderTaskSpec(target *solarv1alpha1.Target, releases []releaseInfo, registry *solarv1alpha1.Registry, bootstrapVersion int64) (solarv1alpha1.RenderTaskSpec, error) {
7✔
1075
        input, err := buildBootstrapInput(target, releases, registry.Spec.TargetPullSecretName)
7✔
1076
        if err != nil {
7✔
1077
                return solarv1alpha1.RenderTaskSpec{}, err
×
1078
        }
×
1079

1080
        releaseNames := make([]string, 0, len(releases))
7✔
1081
        for _, ri := range releases {
16✔
1082
                releaseNames = append(releaseNames, ri.name)
9✔
1083
        }
9✔
1084

1085
        sort.Strings(releaseNames)
7✔
1086

7✔
1087
        chartName := fmt.Sprintf("bootstrap-%s", target.Name)
7✔
1088
        repo := fmt.Sprintf("%s/%s", target.Namespace, chartName)
7✔
1089
        tag := fmt.Sprintf("v0.0.%d", bootstrapVersion)
7✔
1090

7✔
1091
        return solarv1alpha1.RenderTaskSpec{
7✔
1092
                RendererConfig: solarv1alpha1.RendererConfig{
7✔
1093
                        Type: solarv1alpha1.RendererConfigTypeBootstrap,
7✔
1094
                        BootstrapConfig: solarv1alpha1.BootstrapConfig{
7✔
1095
                                Chart: solarv1alpha1.ChartConfig{
7✔
1096
                                        Name:        chartName,
7✔
1097
                                        Description: fmt.Sprintf("Bootstrap of %v", releaseNames),
7✔
1098
                                        Version:     tag,
7✔
1099
                                        AppVersion:  tag,
7✔
1100
                                },
7✔
1101
                                Input: input,
7✔
1102
                        },
7✔
1103
                },
7✔
1104
                Repository:     repo,
7✔
1105
                Tag:            tag,
7✔
1106
                BaseURL:        registry.Spec.Hostname,
7✔
1107
                PushSecretRef:  registry.Spec.SolarSecretRef,
7✔
1108
                OwnerName:      target.Name,
7✔
1109
                OwnerNamespace: target.Namespace,
7✔
1110
                OwnerKind:      "Target",
7✔
1111
        }, nil
7✔
1112
}
1113

1114
// SetupWithManager sets up the controller with the Manager.
1115
func (r *TargetReconciler) SetupWithManager(mgr ctrl.Manager) error {
1✔
1116
        return ctrl.NewControllerManagedBy(mgr).
1✔
1117
                For(&solarv1alpha1.Target{}).
1✔
1118
                Watches(
1✔
1119
                        &solarv1alpha1.ReleaseBinding{},
1✔
1120
                        handler.EnqueueRequestsFromMapFunc(r.mapReleaseBindingToTarget),
1✔
1121
                ).
1✔
1122
                Watches(
1✔
1123
                        &solarv1alpha1.RenderTask{},
1✔
1124
                        handler.EnqueueRequestsFromMapFunc(mapRenderTaskToOwner("Target")),
1✔
1125
                        builder.WithPredicates(renderTaskStatusChangePredicate()),
1✔
1126
                ).
1✔
1127
                Watches(
1✔
1128
                        &solarv1alpha1.Registry{},
1✔
1129
                        handler.EnqueueRequestsFromMapFunc(r.mapRegistryToTargets),
1✔
1130
                ).
1✔
1131
                Watches(
1✔
1132
                        &solarv1alpha1.RegistryBinding{},
1✔
1133
                        handler.EnqueueRequestsFromMapFunc(r.mapRegistryBindingToTarget),
1✔
1134
                ).
1✔
1135
                Watches(
1✔
1136
                        &solarv1alpha1.ReferenceGrant{},
1✔
1137
                        handler.EnqueueRequestsFromMapFunc(r.mapReferenceGrantToTargets),
1✔
1138
                ).
1✔
1139
                Watches(
1✔
1140
                        &solarv1alpha1.Release{},
1✔
1141
                        handler.EnqueueRequestsFromMapFunc(r.mapReleaseToTargets),
1✔
1142
                ).
1✔
1143
                Complete(r)
1✔
1144
}
1✔
1145

1146
// registryGranted checks whether a ReferenceGrant in registryNamespace permits
1147
// fromNamespace to reference the named registry.
1148
func (r *TargetReconciler) registryGranted(ctx context.Context, registryNamespace, fromNamespace string) (bool, error) {
×
1149
        grantList := &solarv1alpha1.ReferenceGrantList{}
×
1150
        if err := r.List(ctx, grantList, client.InNamespace(registryNamespace)); err != nil {
×
1151
                return false, err
×
1152
        }
×
1153
        for i := range grantList.Items {
×
1154
                grant := &grantList.Items[i]
×
1155
                if grantPermitsRegistryAccess(grant, fromNamespace) {
×
1156
                        return true, nil
×
1157
                }
×
1158
        }
1159

1160
        return false, nil
×
1161
}
1162

1163
// grantPermitsRegistryAccess returns true if the ReferenceGrant allows a Target in
1164
// fromNamespace to reference Registry resources in the grant's namespace.
1165
func grantPermitsRegistryAccess(grant *solarv1alpha1.ReferenceGrant, fromNamespace string) bool {
×
1166
        return grantPermits(grant, solarGroup, "Target", fromNamespace, solarGroup, "Registry")
×
1167
}
×
1168

1169
// mapRegistryToTargets maps a Registry event to reconcile requests for all
1170
// Targets that reference it — either in the same namespace or cross-namespace.
1171
func (r *TargetReconciler) mapRegistryToTargets(ctx context.Context, obj client.Object) []reconcile.Request {
31✔
1172
        reg, ok := obj.(*solarv1alpha1.Registry)
31✔
1173
        if !ok {
31✔
1174
                return nil
×
1175
        }
×
1176

1177
        // Same-namespace targets
1178
        targetList := &solarv1alpha1.TargetList{}
31✔
1179
        if err := r.List(ctx, targetList, client.InNamespace(reg.Namespace)); err != nil {
31✔
1180
                ctrl.LoggerFrom(ctx).Error(err, "failed to list Targets for Registry", "registry", reg.Name)
×
1181

×
1182
                return nil
×
1183
        }
×
1184

1185
        var requests []reconcile.Request
31✔
1186
        for _, t := range targetList.Items {
32✔
1187
                if t.Spec.RenderRegistryRef.Name == reg.Name &&
1✔
1188
                        (t.Spec.RenderRegistryNamespace == "" || t.Spec.RenderRegistryNamespace == reg.Namespace) {
1✔
1189
                        requests = append(requests, reconcile.Request{
×
1190
                                NamespacedName: types.NamespacedName{
×
1191
                                        Name:      t.Name,
×
1192
                                        Namespace: t.Namespace,
×
1193
                                },
×
1194
                        })
×
1195
                }
×
1196
        }
1197

1198
        // Cross-namespace targets: find namespaces that have been granted access to
1199
        // registries in reg.Namespace, then check their targets.
1200
        grantList := &solarv1alpha1.ReferenceGrantList{}
31✔
1201
        if err := r.List(ctx, grantList, client.InNamespace(reg.Namespace)); err != nil {
31✔
1202
                ctrl.LoggerFrom(ctx).Error(err, "failed to list ReferenceGrants for cross-namespace Registry mapping")
×
1203
                return requests
×
1204
        }
×
1205

1206
        for i := range grantList.Items {
31✔
1207
                grant := &grantList.Items[i]
×
1208
                if !grantsRegistryResource(grant) {
×
1209
                        continue
×
1210
                }
1211
                for _, from := range grant.Spec.From {
×
1212
                        if from.Kind != "Target" || from.Group != solarGroup {
×
1213
                                continue
×
1214
                        }
1215
                        crossTargets := &solarv1alpha1.TargetList{}
×
1216
                        if err := r.List(ctx, crossTargets, client.InNamespace(from.Namespace)); err != nil {
×
1217
                                ctrl.LoggerFrom(ctx).Error(err, "failed to list cross-namespace Targets", "namespace", from.Namespace)
×
1218
                                continue
×
1219
                        }
1220
                        for _, t := range crossTargets.Items {
×
1221
                                if t.Spec.RenderRegistryRef.Name == reg.Name && t.Spec.RenderRegistryNamespace == reg.Namespace {
×
1222
                                        requests = append(requests, reconcile.Request{
×
1223
                                                NamespacedName: types.NamespacedName{
×
1224
                                                        Name:      t.Name,
×
1225
                                                        Namespace: t.Namespace,
×
1226
                                                },
×
1227
                                        })
×
1228
                                }
×
1229
                        }
1230
                }
1231
        }
1232

1233
        return requests
31✔
1234
}
1235

1236
// buildPullSecretsLookup lists RegistryBindings for the given target, resolves
1237
// each bound Registry, and returns a map from registry hostname to
1238
// targetPullSecretName. Registries without a targetPullSecretName are included
1239
// with an empty string (anonymous pull).
1240
func (r *TargetReconciler) buildPullSecretsLookup(ctx context.Context, target *solarv1alpha1.Target) (map[string]string, error) {
123✔
1241
        rbList := &solarv1alpha1.RegistryBindingList{}
123✔
1242
        if err := r.List(ctx, rbList,
123✔
1243
                client.InNamespace(target.Namespace),
123✔
1244
                client.MatchingFields{indexRegistryBindingTargetName: target.Name},
123✔
1245
        ); err != nil {
123✔
1246
                return nil, err
×
1247
        }
×
1248

1249
        type hostEntry struct {
123✔
1250
                pullSecret  string
123✔
1251
                bindingName string
123✔
1252
        }
123✔
1253

123✔
1254
        lookup := make(map[string]hostEntry, len(rbList.Items))
123✔
1255

123✔
1256
        for _, rb := range rbList.Items {
161✔
1257
                reg := &solarv1alpha1.Registry{}
38✔
1258
                if err := r.Get(ctx, client.ObjectKey{
38✔
1259
                        Name:      rb.Spec.RegistryRef.Name,
38✔
1260
                        Namespace: rb.Namespace,
38✔
1261
                }, reg); err != nil {
48✔
1262
                        return nil, fmt.Errorf("failed to get Registry %s referenced by RegistryBinding %s: %w",
10✔
1263
                                rb.Spec.RegistryRef.Name, rb.Name, err)
10✔
1264
                }
10✔
1265

1266
                host := strings.ToLower(reg.Spec.Hostname)
28✔
1267
                if prev, ok := lookup[host]; ok && prev.pullSecret != reg.Spec.TargetPullSecretName {
38✔
1268
                        return nil, fmt.Errorf("conflicting RegistryBindings for host %q: RegistryBinding %s (pull secret %q) vs RegistryBinding %s (pull secret %q)",
10✔
1269
                                host, prev.bindingName, prev.pullSecret, rb.Name, reg.Spec.TargetPullSecretName)
10✔
1270
                }
10✔
1271

1272
                lookup[host] = hostEntry{pullSecret: reg.Spec.TargetPullSecretName, bindingName: rb.Name}
18✔
1273
        }
1274

1275
        result := make(map[string]string, len(lookup))
103✔
1276
        for host, entry := range lookup {
111✔
1277
                result[host] = entry.pullSecret
8✔
1278
        }
8✔
1279

1280
        return result, nil
103✔
1281
}
1282

1283
// mapRegistryBindingToTarget maps a RegistryBinding event to a reconcile request
1284
// for the referenced Target.
1285
func (r *TargetReconciler) mapRegistryBindingToTarget(ctx context.Context, obj client.Object) []reconcile.Request {
7✔
1286
        rb, ok := obj.(*solarv1alpha1.RegistryBinding)
7✔
1287
        if !ok {
7✔
1288
                return nil
×
1289
        }
×
1290

1291
        if rb.Spec.TargetRef.Name == "" {
7✔
1292
                return nil
×
1293
        }
×
1294

1295
        return []reconcile.Request{
7✔
1296
                {
7✔
1297
                        NamespacedName: types.NamespacedName{
7✔
1298
                                Name:      rb.Spec.TargetRef.Name,
7✔
1299
                                Namespace: rb.Namespace,
7✔
1300
                        },
7✔
1301
                },
7✔
1302
        }
7✔
1303
}
1304

1305
// mapReferenceGrantToTargets enqueues Targets affected by a ReferenceGrant change
1306
// either because the grant controls Registry access (Target → Registry) or because
1307
// it controls ComponentVersion access (Release → ComponentVersion).
1308
func (r *TargetReconciler) mapReferenceGrantToTargets(ctx context.Context, obj client.Object) []reconcile.Request {
14✔
1309
        grant, ok := obj.(*solarv1alpha1.ReferenceGrant)
14✔
1310
        if !ok {
14✔
1311
                return nil
×
1312
        }
×
1313

1314
        var requests []reconcile.Request
14✔
1315

14✔
1316
        if grantsRegistryResource(grant) {
14✔
1317
                for _, from := range grant.Spec.From {
×
1318
                        if from.Kind != "Target" || from.Group != solarGroup {
×
1319
                                continue
×
1320
                        }
1321
                        targets := &solarv1alpha1.TargetList{}
×
1322
                        if err := r.List(ctx, targets, client.InNamespace(from.Namespace)); err != nil {
×
1323
                                ctrl.LoggerFrom(ctx).Error(err, "failed to list Targets for ReferenceGrant mapping", "namespace", from.Namespace)
×
1324
                                continue
×
1325
                        }
1326
                        for _, t := range targets.Items {
×
1327
                                // Enqueue targets that reference a registry specifically in the grant's namespace
×
1328
                                if t.Spec.RenderRegistryNamespace == grant.Namespace {
×
1329
                                        requests = append(requests, reconcile.Request{
×
1330
                                                NamespacedName: types.NamespacedName{
×
1331
                                                        Name:      t.Name,
×
1332
                                                        Namespace: t.Namespace,
×
1333
                                                },
×
1334
                                        })
×
1335
                                }
×
1336
                        }
1337
                }
1338
        }
1339

1340
        if grantsComponentVersionResource(grant) {
20✔
1341
                seen := map[string]struct{}{}
6✔
1342
                for _, from := range grant.Spec.From {
12✔
1343
                        if from.Kind != "Release" || from.Group != solarGroup {
6✔
1344
                                continue
×
1345
                        }
1346
                        bindings := &solarv1alpha1.ReleaseBindingList{}
6✔
1347
                        if err := r.List(ctx, bindings, client.InNamespace(from.Namespace)); err != nil {
6✔
1348
                                ctrl.LoggerFrom(ctx).Error(err, "failed to list ReleaseBindings for ComponentVersion grant mapping", "namespace", from.Namespace)
×
1349
                                continue
×
1350
                        }
1351
                        for _, rb := range bindings.Items {
7✔
1352
                                if rb.Spec.TargetRef.Name == "" {
1✔
1353
                                        continue
×
1354
                                }
1355
                                targetNs := rb.Namespace
1✔
1356
                                if rb.Spec.TargetNamespace != "" {
2✔
1357
                                        targetNs = rb.Spec.TargetNamespace
1✔
1358
                                }
1✔
1359
                                key := targetNs + "/" + rb.Spec.TargetRef.Name
1✔
1360
                                if _, ok := seen[key]; ok {
1✔
1361
                                        continue
×
1362
                                }
1363
                                seen[key] = struct{}{}
1✔
1364
                                requests = append(requests, reconcile.Request{
1✔
1365
                                        NamespacedName: types.NamespacedName{
1✔
1366
                                                Name:      rb.Spec.TargetRef.Name,
1✔
1367
                                                Namespace: targetNs,
1✔
1368
                                        },
1✔
1369
                                })
1✔
1370
                        }
1371
                }
1372
        }
1373

1374
        if grantsReleaseBindingToTargetResource(grant) {
18✔
1375
                // The grant lives in the Target's namespace and authorizes ReleaseBindings from
4✔
1376
                // other namespaces. Enqueue all Targets in the grant's namespace so they pick up
4✔
1377
                // the new or removed cross-namespace ReleaseBindings.
4✔
1378
                targets := &solarv1alpha1.TargetList{}
4✔
1379
                if err := r.List(ctx, targets, client.InNamespace(grant.Namespace)); err != nil {
4✔
1380
                        ctrl.LoggerFrom(ctx).Error(err, "failed to list Targets for ReleaseBinding grant mapping", "namespace", grant.Namespace)
×
1381
                } else {
4✔
1382
                        for _, t := range targets.Items {
8✔
1383
                                requests = append(requests, reconcile.Request{
4✔
1384
                                        NamespacedName: types.NamespacedName{
4✔
1385
                                                Name:      t.Name,
4✔
1386
                                                Namespace: t.Namespace,
4✔
1387
                                        },
4✔
1388
                                })
4✔
1389
                        }
4✔
1390
                }
1391
        }
1392

1393
        return requests
14✔
1394
}
1395

1396
// grantsRegistryResource returns true if the ReferenceGrant includes Registry in its To list.
1397
func grantsRegistryResource(grant *solarv1alpha1.ReferenceGrant) bool {
14✔
1398
        for _, t := range grant.Spec.To {
28✔
1399
                if t.Kind == "Registry" && t.Group == solarGroup {
14✔
1400
                        return true
×
1401
                }
×
1402
        }
1403

1404
        return false
14✔
1405
}
1406

1407
// grantsReleaseBindingToTargetResource returns true if the ReferenceGrant authorizes
1408
// ReleaseBindings in another namespace to reference Targets in the grant's namespace.
1409
func grantsReleaseBindingToTargetResource(grant *solarv1alpha1.ReferenceGrant) bool {
22✔
1410
        hasReleaseBindingFrom := false
22✔
1411
        for _, f := range grant.Spec.From {
44✔
1412
                if f.Kind == "ReleaseBinding" && f.Group == solarGroup {
34✔
1413
                        hasReleaseBindingFrom = true
12✔
1414
                        break
12✔
1415
                }
1416
        }
1417
        if !hasReleaseBindingFrom {
32✔
1418
                return false
10✔
1419
        }
10✔
1420
        for _, t := range grant.Spec.To {
24✔
1421
                if t.Kind == "Target" && t.Group == solarGroup {
24✔
1422
                        return true
12✔
1423
                }
12✔
1424
        }
1425

1426
        return false
×
1427
}
1428

1429
// collectCrossNamespaceReleaseBindings returns ReleaseBindings from other namespaces
1430
// that reference target via spec.targetRef.name + spec.targetNamespace, authorized by
1431
// a ReferenceGrant in target's namespace.
1432
func (r *TargetReconciler) collectCrossNamespaceReleaseBindings(ctx context.Context, target *solarv1alpha1.Target) ([]solarv1alpha1.ReleaseBinding, error) {
103✔
1433
        grantList := &solarv1alpha1.ReferenceGrantList{}
103✔
1434
        if err := r.List(ctx, grantList, client.InNamespace(target.Namespace)); err != nil {
103✔
1435
                return nil, err
×
1436
        }
×
1437

1438
        seen := make(map[string]struct{})
103✔
1439
        var result []solarv1alpha1.ReleaseBinding
103✔
1440
        for i := range grantList.Items {
111✔
1441
                grant := &grantList.Items[i]
8✔
1442
                if !grantsReleaseBindingToTargetResource(grant) {
8✔
1443
                        continue
×
1444
                }
1445
                for _, from := range grant.Spec.From {
16✔
1446
                        if from.Kind != "ReleaseBinding" || from.Group != solarGroup {
8✔
1447
                                continue
×
1448
                        }
1449
                        crossBindings := &solarv1alpha1.ReleaseBindingList{}
8✔
1450
                        if err := r.List(ctx, crossBindings,
8✔
1451
                                client.InNamespace(from.Namespace),
8✔
1452
                                client.MatchingFields{indexReleaseBindingTargetName: target.Name},
8✔
1453
                        ); err != nil {
8✔
1454
                                return nil, err
×
1455
                        }
×
1456
                        for _, rb := range crossBindings.Items {
16✔
1457
                                if rb.Spec.TargetNamespace != target.Namespace {
8✔
1458
                                        continue
×
1459
                                }
1460
                                key := rb.Namespace + "/" + rb.Name
8✔
1461
                                if _, exists := seen[key]; exists {
9✔
1462
                                        continue
1✔
1463
                                }
1464
                                seen[key] = struct{}{}
7✔
1465
                                result = append(result, rb)
7✔
1466
                        }
1467
                }
1468
        }
1469

1470
        return result, nil
103✔
1471
}
1472

1473
// mapReleaseToTargets maps a Release event to reconcile requests for all
1474
// Targets that are bound to the release via ReleaseBindings.
1475
func (r *TargetReconciler) mapReleaseToTargets(ctx context.Context, obj client.Object) []reconcile.Request {
105✔
1476
        rel, ok := obj.(*solarv1alpha1.Release)
105✔
1477
        if !ok {
105✔
1478
                return nil
×
1479
        }
×
1480

1481
        bindingList := &solarv1alpha1.ReleaseBindingList{}
105✔
1482
        if err := r.List(ctx, bindingList,
105✔
1483
                client.InNamespace(rel.Namespace),
105✔
1484
                client.MatchingFields{indexReleaseBindingReleaseName: rel.Name},
105✔
1485
        ); err != nil {
105✔
1486
                ctrl.LoggerFrom(ctx).Error(err, "failed to list ReleaseBindings for Release", "release", rel.Name)
×
1487

×
1488
                return nil
×
1489
        }
×
1490

1491
        seen := map[string]struct{}{}
105✔
1492
        var requests []reconcile.Request
105✔
1493

105✔
1494
        for _, rb := range bindingList.Items {
107✔
1495
                targetNs := rb.Namespace
2✔
1496
                if rb.Spec.TargetNamespace != "" {
2✔
1497
                        targetNs = rb.Spec.TargetNamespace
×
1498
                }
×
1499

1500
                key := targetNs + "/" + rb.Spec.TargetRef.Name
2✔
1501
                if _, ok := seen[key]; ok {
2✔
1502
                        continue
×
1503
                }
1504

1505
                seen[key] = struct{}{}
2✔
1506
                requests = append(requests, reconcile.Request{
2✔
1507
                        NamespacedName: types.NamespacedName{
2✔
1508
                                Name:      rb.Spec.TargetRef.Name,
2✔
1509
                                Namespace: targetNs,
2✔
1510
                        },
2✔
1511
                })
2✔
1512
        }
1513

1514
        return requests
105✔
1515
}
1516

1517
func (r *TargetReconciler) mapReleaseBindingToTarget(_ context.Context, obj client.Object) []reconcile.Request {
39✔
1518
        rb, ok := obj.(*solarv1alpha1.ReleaseBinding)
39✔
1519
        if !ok || rb.Spec.TargetRef.Name == "" {
39✔
1520
                return nil
×
1521
        }
×
1522

1523
        targetNs := rb.Namespace
39✔
1524
        if rb.Spec.TargetNamespace != "" {
48✔
1525
                targetNs = rb.Spec.TargetNamespace
9✔
1526
        }
9✔
1527

1528
        return []reconcile.Request{
39✔
1529
                {
39✔
1530
                        NamespacedName: types.NamespacedName{
39✔
1531
                                Name:      rb.Spec.TargetRef.Name,
39✔
1532
                                Namespace: targetNs,
39✔
1533
                        },
39✔
1534
                },
39✔
1535
        }
39✔
1536
}
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