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

opendefensecloud / solution-arsenal / 27404714757

12 Jun 2026 08:38AM UTC coverage: 74.115% (-0.3%) from 74.416%
27404714757

Pull #575

github

web-flow
Merge 2efe55c6e into 762f047da
Pull Request #575: feat: include e2e tests in CI

2952 of 3983 relevant lines covered (74.11%)

35.41 hits per line

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

72.29
/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
        // WatchNamespace restricts reconciliation to this namespace.
69
        // Should be empty in production (watches all namespaces).
70
        // Intended for use in integration tests only.
71
        WatchNamespace string
72
        // RegistryBindingStrict enables strict registry binding mode.
73
        // When true, rendering fails if a resource's registry host has no
74
        // matching RegistryBinding. When false (default/relaxed), unmatched
75
        // hosts are treated as anonymous pull (no secretRef rendered).
76
        RegistryBindingStrict bool
77
}
78

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

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

287✔
98
        log.V(1).Info("Target is being reconciled", "req", req)
287✔
99

287✔
100
        if r.WatchNamespace != "" && req.Namespace != r.WatchNamespace {
379✔
101
                return ctrl.Result{}, nil
92✔
102
        }
92✔
103

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

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

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

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

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

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

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

145
                return ctrl.Result{}, nil
2✔
146
        }
147

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

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

161
                return ctrl.Result{}, nil
34✔
162
        }
163

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

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

183
                        return ctrl.Result{RequeueAfter: 30 * time.Second}, nil
×
184
                }
185
        }
186

187
        registry := &solarv1alpha1.Registry{}
153✔
188
        if err := r.Get(ctx, client.ObjectKey{
153✔
189
                Name:      target.Spec.RenderRegistryRef.Name,
153✔
190
                Namespace: registryNamespace,
153✔
191
        }, registry); err != nil {
172✔
192
                if apierrors.IsNotFound(err) {
38✔
193
                        if condErr := r.setCondition(ctx, target, ConditionTypeRegistryResolved, metav1.ConditionFalse, "NotFound",
19✔
194
                                "Registry not found: "+target.Spec.RenderRegistryRef.Name); condErr != nil {
19✔
195
                                return ctrl.Result{}, condErr
×
196
                        }
×
197

198
                        return ctrl.Result{RequeueAfter: 30 * time.Second}, nil
19✔
199
                }
200

201
                return ctrl.Result{}, errLogAndWrap(log, err, "failed to get Registry")
×
202
        }
203

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

210
                return ctrl.Result{}, nil
2✔
211
        }
212

213
        if condErr := r.setCondition(ctx, target, ConditionTypeRegistryResolved, metav1.ConditionTrue, "Resolved",
132✔
214
                "Registry resolved: "+registry.Name); condErr != nil {
132✔
215
                return ctrl.Result{}, condErr
×
216
        }
×
217

218
        // Build hostname→targetPullSecretName lookup from RegistryBindings for this target.
219
        pullSecretsByHost, err := r.buildPullSecretsLookup(ctx, target)
132✔
220
        if err != nil {
154✔
221
                if condErr := r.setCondition(ctx, target, ConditionTypeReleasesRendered, metav1.ConditionFalse, "RegistryBindingConflict",
22✔
222
                        err.Error()); condErr != nil {
22✔
223
                        return ctrl.Result{}, condErr
×
224
                }
×
225

226
                return ctrl.Result{}, errLogAndWrap(log, err, "failed to build pull secrets lookup from RegistryBindings")
22✔
227
        }
228

229
        // Collect ReleaseBindings for this target — same namespace first, then cross-namespace via ReferenceGrants.
230
        // Filter on targetNamespace="" to exclude cross-namespace bindings (targetNamespace set) that share the
231
        // target name but point to a target in a different namespace.
232
        bindingList := &solarv1alpha1.ReleaseBindingList{}
110✔
233
        if err := r.List(ctx, bindingList,
110✔
234
                client.InNamespace(target.Namespace),
110✔
235
                client.MatchingFields{
110✔
236
                        indexReleaseBindingTargetName:      target.Name,
110✔
237
                        indexReleaseBindingTargetNamespace: "",
110✔
238
                },
110✔
239
        ); err != nil {
110✔
240
                return ctrl.Result{}, errLogAndWrap(log, err, "failed to list ReleaseBindings")
×
241
        }
×
242

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

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

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

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

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

273
                return ctrl.Result{}, nil
10✔
274
        }
275

276
        // For each bound release, ensure a per-release RenderTask exists
277
        var releases []releaseInfo
100✔
278

100✔
279
        pendingDeps := false
100✔
280

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

×
291
                                continue
×
292
                        }
293

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

297
                cv := &solarv1alpha1.ComponentVersion{}
127✔
298
                cvNamespace := rel.Namespace
127✔
299
                if rel.Spec.ComponentVersionNamespace != "" {
127✔
300
                        cvNamespace = rel.Spec.ComponentVersionNamespace
×
301
                }
×
302

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

×
319
                                continue
×
320
                        }
321
                }
322

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

×
331
                                continue
×
332
                        }
333

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

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

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

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

360
                return ctrl.Result{}, nil
×
361
        }
362

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

100✔
367
        for i, ri := range releases {
217✔
368
                rt := &solarv1alpha1.RenderTask{}
117✔
369
                err := r.Get(ctx, client.ObjectKey{Name: ri.rtName, Namespace: target.Namespace}, rt)
117✔
370

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

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

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

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

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

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

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

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

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

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

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

443
                        return ctrl.Result{}, nil
×
444
                }
445

446
                if apimeta.IsStatusConditionTrue(rt.Status.Conditions, ConditionTypeJobSucceeded) && rt.Status.ChartURL != "" {
150✔
447
                        releases[i].chartURL = rt.Status.ChartURL
43✔
448

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

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

473
                return ctrl.Result{RequeueAfter: 30 * time.Second}, nil
×
474
        }
475

476
        if !allRendered {
150✔
477
                if condErr := r.setCondition(ctx, target, ConditionTypeReleasesRendered, metav1.ConditionFalse, "Pending",
60✔
478
                        "Waiting for release RenderTasks to complete"); condErr != nil {
60✔
479
                        return ctrl.Result{}, condErr
×
480
                }
×
481

482
                return ctrl.Result{RequeueAfter: 30 * time.Second}, nil
60✔
483
        }
484

485
        if condErr := r.setCondition(ctx, target, ConditionTypeReleasesRendered, metav1.ConditionTrue, "AllRendered",
30✔
486
                "All releases rendered successfully"); condErr != nil {
30✔
487
                return ctrl.Result{}, condErr
×
488
        }
×
489

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

30✔
497
        needsNewBootstrap := false
30✔
498

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

513
                existingInput := bootstrapRT.Spec.RendererConfig.BootstrapConfig.Input
25✔
514
                if !apiequality.Semantic.DeepEqual(desiredInput, existingInput) {
27✔
515
                        bootstrapVersion++
2✔
516
                        needsNewBootstrap = true
2✔
517
                }
2✔
518
        }
519

520
        if needsNewBootstrap {
37✔
521
                spec, specErr := r.computeBootstrapRenderTaskSpec(target, releases, registry, bootstrapVersion)
7✔
522
                if specErr != nil {
7✔
523
                        return ctrl.Result{}, errLogAndWrap(log, specErr, "failed to compute bootstrap RenderTask spec")
×
524
                }
×
525

526
                bootstrapRTName = targetRenderTaskName(target.Name, bootstrapVersion)
7✔
527
                bootstrapRT = &solarv1alpha1.RenderTask{
7✔
528
                        ObjectMeta: metav1.ObjectMeta{
7✔
529
                                Name:      bootstrapRTName,
7✔
530
                                Namespace: target.Namespace,
7✔
531
                        },
7✔
532
                        Spec: spec,
7✔
533
                }
7✔
534

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

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

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

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

565
                return ctrl.Result{}, nil
×
566
        }
567

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

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

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

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

609
                return ctrl.Result{}, nil
8✔
610
        }
611

612
        // Still running
613
        return ctrl.Result{}, nil
22✔
614
}
615

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

630
        return nil
403✔
631
}
632

633
func (r *TargetReconciler) setResolvedCondition(ctx context.Context, target *solarv1alpha1.Target, skipped []string) error {
100✔
634
        if len(skipped) == 0 {
190✔
635
                return r.setCondition(ctx, target, ConditionTypeReleasesResolved, metav1.ConditionTrue, "NoConflicts", "")
90✔
636
        }
90✔
637

638
        return r.setCondition(ctx, target, ConditionTypeReleasesResolved, metav1.ConditionTrue, "Resolved", strings.Join(skipped, "; "))
10✔
639
}
640

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

650
        // Step A: uniqueName deduplication.
651
        // When UniqueName is empty, fall back to the parent Component name from the CV.
652
        namedGroups := map[string][]releaseInfo{}
108✔
653

108✔
654
        for i, ri := range releases {
249✔
655
                uname := effectiveUniqueName(ri.release, ri.cv)
141✔
656
                releases[i].uniqueName = uname
141✔
657
                namedGroups[uname] = append(namedGroups[uname], releases[i])
141✔
658
        }
141✔
659

660
        var accepted []releaseInfo
108✔
661

108✔
662
        var skipped []string
108✔
663

108✔
664
        // byPriority sorts releases with highest priority first; bindingKey breaks ties.
108✔
665
        byPriority := func(a, b releaseInfo) bool {
141✔
666
                if a.release.Spec.Priority != b.release.Spec.Priority {
43✔
667
                        return a.release.Spec.Priority > b.release.Spec.Priority
10✔
668
                }
10✔
669

670
                return a.bindingKey < b.bindingKey
23✔
671
        }
672

673
        uniqueNames := make([]string, 0, len(namedGroups))
108✔
674
        for k := range namedGroups {
239✔
675
                uniqueNames = append(uniqueNames, k)
131✔
676
        }
131✔
677

678
        sort.Strings(uniqueNames)
108✔
679

108✔
680
        for _, uniqueName := range uniqueNames {
239✔
681
                group := namedGroups[uniqueName]
131✔
682
                sort.Slice(group, func(i, j int) bool { return byPriority(group[i], group[j]) })
141✔
683

684
                accepted = append(accepted, group[0])
131✔
685

131✔
686
                for _, loser := range group[1:] {
141✔
687
                        skipped = append(skipped, fmt.Sprintf(
10✔
688
                                "binding %s filtered: uniqueName %q conflict, lower priority than %s",
10✔
689
                                loser.bindingKey, uniqueName, group[0].bindingKey,
10✔
690
                        ))
10✔
691
                }
10✔
692
        }
693

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

699
        resolved := make([]releaseInfo, 0, len(accepted))
108✔
700

108✔
701
        for _, ri := range accepted {
239✔
702
                // Parse ri's own anti-affinity selector once; bail early on invalid selector.
131✔
703
                var riSelector labels.Selector
131✔
704
                if ri.release.Spec.AntiAffinity != nil {
138✔
705
                        sel, err := metav1.LabelSelectorAsSelector(ri.release.Spec.AntiAffinity)
7✔
706
                        if err != nil {
8✔
707
                                skipped = append(skipped, fmt.Sprintf(
1✔
708
                                        "binding %s filtered: invalid antiAffinity selector: %v",
1✔
709
                                        ri.bindingKey, err,
1✔
710
                                ))
1✔
711

1✔
712
                                continue
1✔
713
                        }
714

715
                        riSelector = sel
6✔
716
                }
717

718
                // Check both directions: ri's anti-affinity against already-resolved labels,
719
                // and already-resolved anti-affinities against ri's labels.
720
                conflict := ""
130✔
721
                for _, other := range resolved {
153✔
722
                        if riSelector != nil && riSelector.Matches(labels.Set(other.release.Labels)) {
27✔
723
                                conflict = other.bindingKey
4✔
724
                                break
4✔
725
                        }
726

727
                        if other.release.Spec.AntiAffinity != nil {
20✔
728
                                otherSel, err := metav1.LabelSelectorAsSelector(other.release.Spec.AntiAffinity)
1✔
729
                                if err == nil && otherSel.Matches(labels.Set(ri.release.Labels)) {
2✔
730
                                        conflict = other.bindingKey
1✔
731
                                        break
1✔
732
                                }
733
                        }
734
                }
735

736
                if conflict != "" {
135✔
737
                        skipped = append(skipped, fmt.Sprintf(
5✔
738
                                "binding %s filtered: anti-affinity conflict with %s",
5✔
739
                                ri.bindingKey, conflict,
5✔
740
                        ))
5✔
741
                } else {
130✔
742
                        resolved = append(resolved, ri)
125✔
743
                }
125✔
744
        }
745

746
        return resolved, skipped
108✔
747
}
748

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

18✔
755
        rtList := &solarv1alpha1.RenderTaskList{}
18✔
756
        if err := r.List(ctx, rtList,
18✔
757
                client.InNamespace(target.Namespace),
18✔
758
                client.MatchingFields{indexOwnerKind: "Target"},
18✔
759
        ); err != nil {
18✔
760
                return err
×
761
        }
×
762

763
        for i := range rtList.Items {
41✔
764
                rt := &rtList.Items[i]
23✔
765
                if rt.Spec.OwnerName != target.Name || rt.Spec.OwnerNamespace != target.Namespace {
23✔
766
                        continue
×
767
                }
768

769
                if _, current := currentRTNames[rt.Name]; current {
43✔
770
                        continue
20✔
771
                }
772

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

778
                r.Recorder.Eventf(target, nil, corev1.EventTypeNormal, "Deleted", "Delete",
3✔
779
                        "Deleted stale RenderTask %s", rt.Name)
3✔
780
        }
781

782
        return nil
18✔
783
}
784

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

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

803
        return nil
3✔
804
}
805

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

18✔
811
        bindingList := &solarv1alpha1.RenderBindingList{}
18✔
812
        if err := r.List(ctx, bindingList,
18✔
813
                client.InNamespace(target.Namespace),
18✔
814
                client.MatchingFields{indexOwnerKind: "Target"},
18✔
815
        ); err != nil {
18✔
816
                return err
×
817
        }
×
818

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

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

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

835
        return nil
18✔
836
}
837

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

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

858
        return nil
3✔
859
}
860

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

877
                return nil
40✔
878
        } else if !apierrors.IsNotFound(err) {
11✔
879
                return err
×
880
        }
×
881

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

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

902
        return nil
11✔
903
}
904

905
// ensureRenderBinding creates a RenderBinding linking this Target to the named
906
// RenderArtifact if one does not already exist. Idempotent.
907
func (r *TargetReconciler) ensureRenderBinding(ctx context.Context, target *solarv1alpha1.Target, artifactName, bindingName string) error {
51✔
908
        binding := &solarv1alpha1.RenderBinding{}
51✔
909
        if err := r.Get(ctx, client.ObjectKey{Name: bindingName, Namespace: target.Namespace}, binding); err == nil {
91✔
910
                return nil
40✔
911
        } else if !apierrors.IsNotFound(err) {
51✔
912
                return err
×
913
        }
×
914

915
        binding = &solarv1alpha1.RenderBinding{
11✔
916
                ObjectMeta: metav1.ObjectMeta{
11✔
917
                        Name:      bindingName,
11✔
918
                        Namespace: target.Namespace,
11✔
919
                },
11✔
920
                Spec: solarv1alpha1.RenderBindingSpec{
11✔
921
                        RenderArtifactRef: corev1.LocalObjectReference{Name: artifactName},
11✔
922
                        OwnerKind:         "Target",
11✔
923
                        OwnerName:         target.Name,
11✔
924
                        OwnerNamespace:    target.Namespace,
11✔
925
                },
11✔
926
        }
11✔
927

11✔
928
        if err := r.Create(ctx, binding); err != nil && !apierrors.IsAlreadyExists(err) {
11✔
929
                return err
×
930
        }
×
931

932
        return nil
11✔
933
}
934

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

117✔
939
        var targetNamespace string
117✔
940
        if rel.Spec.TargetNamespace != nil {
234✔
941
                targetNamespace = *rel.Spec.TargetNamespace
117✔
942
        }
117✔
943

944
        resolvedResources, err := resolveResources(cv.Spec.Resources, pullSecretsByHost, r.RegistryBindingStrict)
117✔
945
        if err != nil {
127✔
946
                return solarv1alpha1.RenderTaskSpec{}, fmt.Errorf("release %s: %w", rel.Name, err)
10✔
947
        }
10✔
948

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

107✔
955
        return solarv1alpha1.RenderTaskSpec{
107✔
956
                RendererConfig: solarv1alpha1.RendererConfig{
107✔
957
                        Type: solarv1alpha1.RendererConfigTypeRelease,
107✔
958
                        ReleaseConfig: solarv1alpha1.ReleaseConfig{
107✔
959
                                Chart: solarv1alpha1.ChartConfig{
107✔
960
                                        Name:        chartName,
107✔
961
                                        Description: fmt.Sprintf("Release of %s", rel.Spec.ComponentVersionRef.Name),
107✔
962
                                        Version:     tag,
107✔
963
                                        AppVersion:  tag,
107✔
964
                                },
107✔
965
                                Input: solarv1alpha1.ReleaseInput{
107✔
966
                                        Component:  solarv1alpha1.ReleaseComponent{Name: cv.Spec.ComponentRef.Name},
107✔
967
                                        Resources:  resolvedResources,
107✔
968
                                        Entrypoint: cv.Spec.Entrypoint,
107✔
969
                                },
107✔
970
                                Values:          rel.Spec.Values,
107✔
971
                                TargetNamespace: targetNamespace,
107✔
972
                        },
107✔
973
                },
107✔
974
                Repository:     repo,
107✔
975
                Tag:            tag,
107✔
976
                BaseURL:        registry.Spec.Hostname,
107✔
977
                PushSecretRef:  registry.Spec.SolarSecretRef,
107✔
978
                FailedJobTTL:   rel.Spec.FailedJobTTL,
107✔
979
                OwnerName:      target.Name,
107✔
980
                OwnerNamespace: target.Namespace,
107✔
981
                OwnerKind:      "Target",
107✔
982
        }, nil
107✔
983
}
984

985
// buildBootstrapInput constructs the desired BootstrapInput from the current
986
// target and resolved releases. Used for both comparison and spec construction.
987
func buildBootstrapInput(target *solarv1alpha1.Target, releases []releaseInfo, renderRegistryPullSecret string) (solarv1alpha1.BootstrapInput, error) {
34✔
988
        resolvedReleases := map[string]solarv1alpha1.ResolvedResourceAccess{}
34✔
989

34✔
990
        for _, ri := range releases {
80✔
991
                if ri.uniqueName == "" {
47✔
992
                        return solarv1alpha1.BootstrapInput{}, fmt.Errorf("release %q has empty uniqueName; resolveReleaseConflicts must run before buildBootstrapInput", ri.name)
1✔
993
                }
1✔
994

995
                ref, err := ociname.ParseReference(ri.chartURL)
45✔
996
                if err != nil {
45✔
997
                        return solarv1alpha1.BootstrapInput{}, fmt.Errorf("failed to parse chartURL %s: %w", ri.chartURL, err)
×
998
                }
×
999

1000
                repo, err := url.JoinPath(ref.Context().RegistryStr(), ref.Context().RepositoryStr())
45✔
1001
                if err != nil {
45✔
1002
                        return solarv1alpha1.BootstrapInput{}, err
×
1003
                }
×
1004

1005
                resolvedReleases[ri.uniqueName] = solarv1alpha1.ResolvedResourceAccess{
45✔
1006
                        Repository:     strings.TrimPrefix(repo, "oci://"),
45✔
1007
                        Tag:            ref.Identifier(),
45✔
1008
                        PullSecretName: renderRegistryPullSecret,
45✔
1009
                }
45✔
1010
        }
1011

1012
        return solarv1alpha1.BootstrapInput{
33✔
1013
                Releases: resolvedReleases,
33✔
1014
                Userdata: target.Spec.Userdata,
33✔
1015
        }, nil
33✔
1016
}
1017

1018
func (r *TargetReconciler) computeBootstrapRenderTaskSpec(target *solarv1alpha1.Target, releases []releaseInfo, registry *solarv1alpha1.Registry, bootstrapVersion int64) (solarv1alpha1.RenderTaskSpec, error) {
7✔
1019
        input, err := buildBootstrapInput(target, releases, registry.Spec.TargetPullSecretName)
7✔
1020
        if err != nil {
7✔
1021
                return solarv1alpha1.RenderTaskSpec{}, err
×
1022
        }
×
1023

1024
        releaseNames := make([]string, 0, len(releases))
7✔
1025
        for _, ri := range releases {
16✔
1026
                releaseNames = append(releaseNames, ri.name)
9✔
1027
        }
9✔
1028

1029
        sort.Strings(releaseNames)
7✔
1030

7✔
1031
        chartName := fmt.Sprintf("bootstrap-%s", target.Name)
7✔
1032
        repo := fmt.Sprintf("%s/%s", target.Namespace, chartName)
7✔
1033
        tag := fmt.Sprintf("v0.0.%d", bootstrapVersion)
7✔
1034

7✔
1035
        return solarv1alpha1.RenderTaskSpec{
7✔
1036
                RendererConfig: solarv1alpha1.RendererConfig{
7✔
1037
                        Type: solarv1alpha1.RendererConfigTypeBootstrap,
7✔
1038
                        BootstrapConfig: solarv1alpha1.BootstrapConfig{
7✔
1039
                                Chart: solarv1alpha1.ChartConfig{
7✔
1040
                                        Name:        chartName,
7✔
1041
                                        Description: fmt.Sprintf("Bootstrap of %v", releaseNames),
7✔
1042
                                        Version:     tag,
7✔
1043
                                        AppVersion:  tag,
7✔
1044
                                },
7✔
1045
                                Input: input,
7✔
1046
                        },
7✔
1047
                },
7✔
1048
                Repository:     repo,
7✔
1049
                Tag:            tag,
7✔
1050
                BaseURL:        registry.Spec.Hostname,
7✔
1051
                PushSecretRef:  registry.Spec.SolarSecretRef,
7✔
1052
                OwnerName:      target.Name,
7✔
1053
                OwnerNamespace: target.Namespace,
7✔
1054
                OwnerKind:      "Target",
7✔
1055
        }, nil
7✔
1056
}
1057

1058
// SetupWithManager sets up the controller with the Manager.
1059
func (r *TargetReconciler) SetupWithManager(mgr ctrl.Manager) error {
1✔
1060
        return ctrl.NewControllerManagedBy(mgr).
1✔
1061
                For(&solarv1alpha1.Target{}).
1✔
1062
                Watches(
1✔
1063
                        &solarv1alpha1.ReleaseBinding{},
1✔
1064
                        handler.EnqueueRequestsFromMapFunc(r.mapReleaseBindingToTarget),
1✔
1065
                ).
1✔
1066
                Watches(
1✔
1067
                        &solarv1alpha1.RenderTask{},
1✔
1068
                        handler.EnqueueRequestsFromMapFunc(mapRenderTaskToOwner("Target")),
1✔
1069
                        builder.WithPredicates(renderTaskStatusChangePredicate()),
1✔
1070
                ).
1✔
1071
                Watches(
1✔
1072
                        &solarv1alpha1.Registry{},
1✔
1073
                        handler.EnqueueRequestsFromMapFunc(r.mapRegistryToTargets),
1✔
1074
                ).
1✔
1075
                Watches(
1✔
1076
                        &solarv1alpha1.RegistryBinding{},
1✔
1077
                        handler.EnqueueRequestsFromMapFunc(r.mapRegistryBindingToTarget),
1✔
1078
                ).
1✔
1079
                Watches(
1✔
1080
                        &solarv1alpha1.ReferenceGrant{},
1✔
1081
                        handler.EnqueueRequestsFromMapFunc(r.mapReferenceGrantToTargets),
1✔
1082
                ).
1✔
1083
                Watches(
1✔
1084
                        &solarv1alpha1.Release{},
1✔
1085
                        handler.EnqueueRequestsFromMapFunc(r.mapReleaseToTargets),
1✔
1086
                ).
1✔
1087
                Complete(r)
1✔
1088
}
1✔
1089

1090
// registryGranted checks whether a ReferenceGrant in registryNamespace permits
1091
// fromNamespace to reference the named registry.
1092
func (r *TargetReconciler) registryGranted(ctx context.Context, registryNamespace, fromNamespace string) (bool, error) {
×
1093
        grantList := &solarv1alpha1.ReferenceGrantList{}
×
1094
        if err := r.List(ctx, grantList, client.InNamespace(registryNamespace)); err != nil {
×
1095
                return false, err
×
1096
        }
×
1097
        for i := range grantList.Items {
×
1098
                grant := &grantList.Items[i]
×
1099
                if grantPermitsRegistryAccess(grant, fromNamespace) {
×
1100
                        return true, nil
×
1101
                }
×
1102
        }
1103

1104
        return false, nil
×
1105
}
1106

1107
// grantPermitsRegistryAccess returns true if the ReferenceGrant allows a Target in
1108
// fromNamespace to reference Registry resources in the grant's namespace.
1109
func grantPermitsRegistryAccess(grant *solarv1alpha1.ReferenceGrant, fromNamespace string) bool {
×
1110
        return grantPermits(grant, solarGroup, "Target", fromNamespace, solarGroup, "Registry")
×
1111
}
×
1112

1113
// mapRegistryToTargets maps a Registry event to reconcile requests for all
1114
// Targets that reference it — either in the same namespace or cross-namespace.
1115
func (r *TargetReconciler) mapRegistryToTargets(ctx context.Context, obj client.Object) []reconcile.Request {
31✔
1116
        reg, ok := obj.(*solarv1alpha1.Registry)
31✔
1117
        if !ok {
31✔
1118
                return nil
×
1119
        }
×
1120

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

×
1126
                return nil
×
1127
        }
×
1128

1129
        var requests []reconcile.Request
31✔
1130
        for _, t := range targetList.Items {
32✔
1131
                if t.Spec.RenderRegistryRef.Name == reg.Name &&
1✔
1132
                        (t.Spec.RenderRegistryNamespace == "" || t.Spec.RenderRegistryNamespace == reg.Namespace) {
1✔
1133
                        requests = append(requests, reconcile.Request{
×
1134
                                NamespacedName: types.NamespacedName{
×
1135
                                        Name:      t.Name,
×
1136
                                        Namespace: t.Namespace,
×
1137
                                },
×
1138
                        })
×
1139
                }
×
1140
        }
1141

1142
        // Cross-namespace targets: find namespaces that have been granted access to
1143
        // registries in reg.Namespace, then check their targets.
1144
        grantList := &solarv1alpha1.ReferenceGrantList{}
31✔
1145
        if err := r.List(ctx, grantList, client.InNamespace(reg.Namespace)); err != nil {
31✔
1146
                ctrl.LoggerFrom(ctx).Error(err, "failed to list ReferenceGrants for cross-namespace Registry mapping")
×
1147
                return requests
×
1148
        }
×
1149

1150
        for i := range grantList.Items {
31✔
1151
                grant := &grantList.Items[i]
×
1152
                if !grantsRegistryResource(grant) {
×
1153
                        continue
×
1154
                }
1155
                for _, from := range grant.Spec.From {
×
1156
                        if from.Kind != "Target" || from.Group != solarGroup {
×
1157
                                continue
×
1158
                        }
1159
                        crossTargets := &solarv1alpha1.TargetList{}
×
1160
                        if err := r.List(ctx, crossTargets, client.InNamespace(from.Namespace)); err != nil {
×
1161
                                ctrl.LoggerFrom(ctx).Error(err, "failed to list cross-namespace Targets", "namespace", from.Namespace)
×
1162
                                continue
×
1163
                        }
1164
                        for _, t := range crossTargets.Items {
×
1165
                                if t.Spec.RenderRegistryRef.Name == reg.Name && t.Spec.RenderRegistryNamespace == reg.Namespace {
×
1166
                                        requests = append(requests, reconcile.Request{
×
1167
                                                NamespacedName: types.NamespacedName{
×
1168
                                                        Name:      t.Name,
×
1169
                                                        Namespace: t.Namespace,
×
1170
                                                },
×
1171
                                        })
×
1172
                                }
×
1173
                        }
1174
                }
1175
        }
1176

1177
        return requests
31✔
1178
}
1179

1180
// buildPullSecretsLookup lists RegistryBindings for the given target, resolves
1181
// each bound Registry, and returns a map from registry hostname to
1182
// targetPullSecretName. Registries without a targetPullSecretName are included
1183
// with an empty string (anonymous pull).
1184
func (r *TargetReconciler) buildPullSecretsLookup(ctx context.Context, target *solarv1alpha1.Target) (map[string]string, error) {
132✔
1185
        rbList := &solarv1alpha1.RegistryBindingList{}
132✔
1186
        if err := r.List(ctx, rbList,
132✔
1187
                client.InNamespace(target.Namespace),
132✔
1188
                client.MatchingFields{indexRegistryBindingTargetName: target.Name},
132✔
1189
        ); err != nil {
132✔
1190
                return nil, err
×
1191
        }
×
1192

1193
        type hostEntry struct {
132✔
1194
                pullSecret  string
132✔
1195
                bindingName string
132✔
1196
        }
132✔
1197

132✔
1198
        lookup := make(map[string]hostEntry, len(rbList.Items))
132✔
1199

132✔
1200
        for _, rb := range rbList.Items {
175✔
1201
                reg := &solarv1alpha1.Registry{}
43✔
1202
                if err := r.Get(ctx, client.ObjectKey{
43✔
1203
                        Name:      rb.Spec.RegistryRef.Name,
43✔
1204
                        Namespace: rb.Namespace,
43✔
1205
                }, reg); err != nil {
53✔
1206
                        return nil, fmt.Errorf("failed to get Registry %s referenced by RegistryBinding %s: %w",
10✔
1207
                                rb.Spec.RegistryRef.Name, rb.Name, err)
10✔
1208
                }
10✔
1209

1210
                host := strings.ToLower(reg.Spec.Hostname)
33✔
1211
                if prev, ok := lookup[host]; ok && prev.pullSecret != reg.Spec.TargetPullSecretName {
45✔
1212
                        return nil, fmt.Errorf("conflicting RegistryBindings for host %q: RegistryBinding %s (pull secret %q) vs RegistryBinding %s (pull secret %q)",
12✔
1213
                                host, prev.bindingName, prev.pullSecret, rb.Name, reg.Spec.TargetPullSecretName)
12✔
1214
                }
12✔
1215

1216
                lookup[host] = hostEntry{pullSecret: reg.Spec.TargetPullSecretName, bindingName: rb.Name}
21✔
1217
        }
1218

1219
        result := make(map[string]string, len(lookup))
110✔
1220
        for host, entry := range lookup {
119✔
1221
                result[host] = entry.pullSecret
9✔
1222
        }
9✔
1223

1224
        return result, nil
110✔
1225
}
1226

1227
// mapRegistryBindingToTarget maps a RegistryBinding event to a reconcile request
1228
// for the referenced Target.
1229
func (r *TargetReconciler) mapRegistryBindingToTarget(ctx context.Context, obj client.Object) []reconcile.Request {
7✔
1230
        rb, ok := obj.(*solarv1alpha1.RegistryBinding)
7✔
1231
        if !ok {
7✔
1232
                return nil
×
1233
        }
×
1234

1235
        if rb.Spec.TargetRef.Name == "" {
7✔
1236
                return nil
×
1237
        }
×
1238

1239
        return []reconcile.Request{
7✔
1240
                {
7✔
1241
                        NamespacedName: types.NamespacedName{
7✔
1242
                                Name:      rb.Spec.TargetRef.Name,
7✔
1243
                                Namespace: rb.Namespace,
7✔
1244
                        },
7✔
1245
                },
7✔
1246
        }
7✔
1247
}
1248

1249
// mapReferenceGrantToTargets enqueues Targets affected by a ReferenceGrant change
1250
// either because the grant controls Registry access (Target → Registry) or because
1251
// it controls ComponentVersion access (Release → ComponentVersion).
1252
func (r *TargetReconciler) mapReferenceGrantToTargets(ctx context.Context, obj client.Object) []reconcile.Request {
13✔
1253
        grant, ok := obj.(*solarv1alpha1.ReferenceGrant)
13✔
1254
        if !ok {
13✔
1255
                return nil
×
1256
        }
×
1257

1258
        var requests []reconcile.Request
13✔
1259

13✔
1260
        if grantsRegistryResource(grant) {
13✔
1261
                for _, from := range grant.Spec.From {
×
1262
                        if from.Kind != "Target" || from.Group != solarGroup {
×
1263
                                continue
×
1264
                        }
1265
                        targets := &solarv1alpha1.TargetList{}
×
1266
                        if err := r.List(ctx, targets, client.InNamespace(from.Namespace)); err != nil {
×
1267
                                ctrl.LoggerFrom(ctx).Error(err, "failed to list Targets for ReferenceGrant mapping", "namespace", from.Namespace)
×
1268
                                continue
×
1269
                        }
1270
                        for _, t := range targets.Items {
×
1271
                                // Enqueue targets that reference a registry specifically in the grant's namespace
×
1272
                                if t.Spec.RenderRegistryNamespace == grant.Namespace {
×
1273
                                        requests = append(requests, reconcile.Request{
×
1274
                                                NamespacedName: types.NamespacedName{
×
1275
                                                        Name:      t.Name,
×
1276
                                                        Namespace: t.Namespace,
×
1277
                                                },
×
1278
                                        })
×
1279
                                }
×
1280
                        }
1281
                }
1282
        }
1283

1284
        if grantsComponentVersionResource(grant) {
18✔
1285
                seen := map[string]struct{}{}
5✔
1286
                for _, from := range grant.Spec.From {
10✔
1287
                        if from.Kind != "Release" || from.Group != solarGroup {
5✔
1288
                                continue
×
1289
                        }
1290
                        bindings := &solarv1alpha1.ReleaseBindingList{}
5✔
1291
                        if err := r.List(ctx, bindings, client.InNamespace(from.Namespace)); err != nil {
5✔
1292
                                ctrl.LoggerFrom(ctx).Error(err, "failed to list ReleaseBindings for ComponentVersion grant mapping", "namespace", from.Namespace)
×
1293
                                continue
×
1294
                        }
1295
                        for _, rb := range bindings.Items {
6✔
1296
                                if rb.Spec.TargetRef.Name == "" {
1✔
1297
                                        continue
×
1298
                                }
1299
                                targetNs := rb.Namespace
1✔
1300
                                if rb.Spec.TargetNamespace != "" {
2✔
1301
                                        targetNs = rb.Spec.TargetNamespace
1✔
1302
                                }
1✔
1303
                                key := targetNs + "/" + rb.Spec.TargetRef.Name
1✔
1304
                                if _, ok := seen[key]; ok {
1✔
1305
                                        continue
×
1306
                                }
1307
                                seen[key] = struct{}{}
1✔
1308
                                requests = append(requests, reconcile.Request{
1✔
1309
                                        NamespacedName: types.NamespacedName{
1✔
1310
                                                Name:      rb.Spec.TargetRef.Name,
1✔
1311
                                                Namespace: targetNs,
1✔
1312
                                        },
1✔
1313
                                })
1✔
1314
                        }
1315
                }
1316
        }
1317

1318
        if grantsReleaseBindingToTargetResource(grant) {
17✔
1319
                // The grant lives in the Target's namespace and authorizes ReleaseBindings from
4✔
1320
                // other namespaces. Enqueue all Targets in the grant's namespace so they pick up
4✔
1321
                // the new or removed cross-namespace ReleaseBindings.
4✔
1322
                targets := &solarv1alpha1.TargetList{}
4✔
1323
                if err := r.List(ctx, targets, client.InNamespace(grant.Namespace)); err != nil {
4✔
1324
                        ctrl.LoggerFrom(ctx).Error(err, "failed to list Targets for ReleaseBinding grant mapping", "namespace", grant.Namespace)
×
1325
                } else {
4✔
1326
                        for _, t := range targets.Items {
8✔
1327
                                requests = append(requests, reconcile.Request{
4✔
1328
                                        NamespacedName: types.NamespacedName{
4✔
1329
                                                Name:      t.Name,
4✔
1330
                                                Namespace: t.Namespace,
4✔
1331
                                        },
4✔
1332
                                })
4✔
1333
                        }
4✔
1334
                }
1335
        }
1336

1337
        return requests
13✔
1338
}
1339

1340
// grantsRegistryResource returns true if the ReferenceGrant includes Registry in its To list.
1341
func grantsRegistryResource(grant *solarv1alpha1.ReferenceGrant) bool {
13✔
1342
        for _, t := range grant.Spec.To {
26✔
1343
                if t.Kind == "Registry" && t.Group == solarGroup {
13✔
1344
                        return true
×
1345
                }
×
1346
        }
1347

1348
        return false
13✔
1349
}
1350

1351
// grantsReleaseBindingToTargetResource returns true if the ReferenceGrant authorizes
1352
// ReleaseBindings in another namespace to reference Targets in the grant's namespace.
1353
func grantsReleaseBindingToTargetResource(grant *solarv1alpha1.ReferenceGrant) bool {
24✔
1354
        hasReleaseBindingFrom := false
24✔
1355
        for _, f := range grant.Spec.From {
48✔
1356
                if f.Kind == "ReleaseBinding" && f.Group == solarGroup {
39✔
1357
                        hasReleaseBindingFrom = true
15✔
1358
                        break
15✔
1359
                }
1360
        }
1361
        if !hasReleaseBindingFrom {
33✔
1362
                return false
9✔
1363
        }
9✔
1364
        for _, t := range grant.Spec.To {
30✔
1365
                if t.Kind == "Target" && t.Group == solarGroup {
30✔
1366
                        return true
15✔
1367
                }
15✔
1368
        }
1369

1370
        return false
×
1371
}
1372

1373
// collectCrossNamespaceReleaseBindings returns ReleaseBindings from other namespaces
1374
// that reference target via spec.targetRef.name + spec.targetNamespace, authorized by
1375
// a ReferenceGrant in target's namespace.
1376
func (r *TargetReconciler) collectCrossNamespaceReleaseBindings(ctx context.Context, target *solarv1alpha1.Target) ([]solarv1alpha1.ReleaseBinding, error) {
110✔
1377
        grantList := &solarv1alpha1.ReferenceGrantList{}
110✔
1378
        if err := r.List(ctx, grantList, client.InNamespace(target.Namespace)); err != nil {
110✔
1379
                return nil, err
×
1380
        }
×
1381

1382
        seen := make(map[string]struct{})
110✔
1383
        var result []solarv1alpha1.ReleaseBinding
110✔
1384
        for i := range grantList.Items {
121✔
1385
                grant := &grantList.Items[i]
11✔
1386
                if !grantsReleaseBindingToTargetResource(grant) {
11✔
1387
                        continue
×
1388
                }
1389
                for _, from := range grant.Spec.From {
22✔
1390
                        if from.Kind != "ReleaseBinding" || from.Group != solarGroup {
11✔
1391
                                continue
×
1392
                        }
1393
                        crossBindings := &solarv1alpha1.ReleaseBindingList{}
11✔
1394
                        if err := r.List(ctx, crossBindings,
11✔
1395
                                client.InNamespace(from.Namespace),
11✔
1396
                                client.MatchingFields{indexReleaseBindingTargetName: target.Name},
11✔
1397
                        ); err != nil {
11✔
1398
                                return nil, err
×
1399
                        }
×
1400
                        for _, rb := range crossBindings.Items {
19✔
1401
                                if rb.Spec.TargetNamespace != target.Namespace {
8✔
1402
                                        continue
×
1403
                                }
1404
                                key := rb.Namespace + "/" + rb.Name
8✔
1405
                                if _, exists := seen[key]; exists {
9✔
1406
                                        continue
1✔
1407
                                }
1408
                                seen[key] = struct{}{}
7✔
1409
                                result = append(result, rb)
7✔
1410
                        }
1411
                }
1412
        }
1413

1414
        return result, nil
110✔
1415
}
1416

1417
// mapReleaseToTargets maps a Release event to reconcile requests for all
1418
// Targets that are bound to the release via ReleaseBindings.
1419
func (r *TargetReconciler) mapReleaseToTargets(ctx context.Context, obj client.Object) []reconcile.Request {
105✔
1420
        rel, ok := obj.(*solarv1alpha1.Release)
105✔
1421
        if !ok {
105✔
1422
                return nil
×
1423
        }
×
1424

1425
        bindingList := &solarv1alpha1.ReleaseBindingList{}
105✔
1426
        if err := r.List(ctx, bindingList,
105✔
1427
                client.InNamespace(rel.Namespace),
105✔
1428
                client.MatchingFields{indexReleaseBindingReleaseName: rel.Name},
105✔
1429
        ); err != nil {
105✔
1430
                ctrl.LoggerFrom(ctx).Error(err, "failed to list ReleaseBindings for Release", "release", rel.Name)
×
1431

×
1432
                return nil
×
1433
        }
×
1434

1435
        seen := map[string]struct{}{}
105✔
1436
        var requests []reconcile.Request
105✔
1437

105✔
1438
        for _, rb := range bindingList.Items {
107✔
1439
                targetNs := rb.Namespace
2✔
1440
                if rb.Spec.TargetNamespace != "" {
2✔
1441
                        targetNs = rb.Spec.TargetNamespace
×
1442
                }
×
1443

1444
                key := targetNs + "/" + rb.Spec.TargetRef.Name
2✔
1445
                if _, ok := seen[key]; ok {
2✔
1446
                        continue
×
1447
                }
1448

1449
                seen[key] = struct{}{}
2✔
1450
                requests = append(requests, reconcile.Request{
2✔
1451
                        NamespacedName: types.NamespacedName{
2✔
1452
                                Name:      rb.Spec.TargetRef.Name,
2✔
1453
                                Namespace: targetNs,
2✔
1454
                        },
2✔
1455
                })
2✔
1456
        }
1457

1458
        return requests
105✔
1459
}
1460

1461
func (r *TargetReconciler) mapReleaseBindingToTarget(_ context.Context, obj client.Object) []reconcile.Request {
39✔
1462
        rb, ok := obj.(*solarv1alpha1.ReleaseBinding)
39✔
1463
        if !ok || rb.Spec.TargetRef.Name == "" {
39✔
1464
                return nil
×
1465
        }
×
1466

1467
        targetNs := rb.Namespace
39✔
1468
        if rb.Spec.TargetNamespace != "" {
48✔
1469
                targetNs = rb.Spec.TargetNamespace
9✔
1470
        }
9✔
1471

1472
        return []reconcile.Request{
39✔
1473
                {
39✔
1474
                        NamespacedName: types.NamespacedName{
39✔
1475
                                Name:      rb.Spec.TargetRef.Name,
39✔
1476
                                Namespace: targetNs,
39✔
1477
                        },
39✔
1478
                },
39✔
1479
        }
39✔
1480
}
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