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

opendefensecloud / solution-arsenal / 22721885493

05 Mar 2026 02:13PM UTC coverage: 70.576%. First build
22721885493

Pull #195

github

web-flow
Merge 65ffe8a40 into b03730095
Pull Request #195: Reference Credentials from Secret

236 of 261 new or added lines in 5 files covered. (90.42%)

1998 of 2831 relevant lines covered (70.58%)

20.46 hits per line

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

88.12
/pkg/controller/rendertask_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
        "encoding/json"
9
        "fmt"
10
        "slices"
11
        "strings"
12
        "time"
13

14
        batchv1 "k8s.io/api/batch/v1"
15
        corev1 "k8s.io/api/core/v1"
16
        apierrors "k8s.io/apimachinery/pkg/api/errors"
17
        apimeta "k8s.io/apimachinery/pkg/api/meta"
18
        metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
19
        "k8s.io/apimachinery/pkg/runtime"
20
        "k8s.io/client-go/tools/events"
21
        "k8s.io/utils/ptr"
22
        ctrl "sigs.k8s.io/controller-runtime"
23
        "sigs.k8s.io/controller-runtime/pkg/client"
24
        "sigs.k8s.io/controller-runtime/pkg/controller/controllerutil"
25

26
        solarv1alpha1 "go.opendefense.cloud/solar/api/solar/v1alpha1"
27
)
28

29
const (
30
        renderTaskFinalizer = "solar.opendefense.cloud/rendertask-finalizer"
31

32
        annotationJobName    = "solar.opendefense.cloud/job-name"
33
        annotationSecretName = "solar.opendefense.cloud/secret-name"
34

35
        // Condition types
36
        ConditionTypeJobScheduled = "JobScheduled"
37
        ConditionTypeJobSucceeded = "JobSucceeded"
38
        ConditionTypeJobFailed    = "JobFailed"
39

40
        ConditionTypeTaskCompleted = "TaskCompleted"
41
        ConditionTypeTaskFailed    = "TaskFailed"
42
)
43

44
// RenderTaskReconciler reconciles a RenderTask object
45
type RenderTaskReconciler struct {
46
        client.Client
47
        Scheme          *runtime.Scheme
48
        Recorder        events.EventRecorder
49
        RendererImage   string
50
        RendererCommand string
51
        RendererArgs    []string
52
        PushSecretRef   *corev1.SecretReference
53
        BaseURL         string
54
}
55

56
//+kubebuilder:rbac:groups=solar.opendefense.cloud,resources=rendertasks,verbs=get;list;watch;create;update;patch;delete
57
//+kubebuilder:rbac:groups=solar.opendefense.cloud,resources=rendertasks/status,verbs=get;update;patch
58
//+kubebuilder:rbac:groups=solar.opendefense.cloud,resources=rendertasks/finalizers,verbs=update
59
//+kubebuilder:rbac:groups=batch,resources=jobs,verbs=get;list;watch;create;update;patch;delete
60
//+kubebuilder:rbac:groups="",resources=secrets,verbs=get;list;watch;create;update;patch;delete
61
//+kubebuilder:rbac:groups=core,resources=events,verbs=create;patch
62

63
// Reconcile moves the current state of the cluster closer to the desired state
64
func (r *RenderTaskReconciler) Reconcile(ctx context.Context, req ctrl.Request) (ctrl.Result, error) {
460✔
65
        log := ctrl.LoggerFrom(ctx)
460✔
66
        ctrlResult := ctrl.Result{}
460✔
67

460✔
68
        log.V(1).Info("RenderTask is being reconciled", "req", req)
460✔
69

460✔
70
        // Fetch the RenderTask instance
460✔
71
        res := &solarv1alpha1.RenderTask{}
460✔
72
        if err := r.Get(ctx, req.NamespacedName, res); err != nil {
474✔
73
                if apierrors.IsNotFound(err) {
28✔
74
                        // Object not found, return. Created objects are automatically garbage collected.
14✔
75
                        return ctrlResult, nil
14✔
76
                }
14✔
77

78
                return ctrlResult, errLogAndWrap(log, err, "failed to get object")
×
79
        }
80

81
        // Handle deletion: cleanup job and secret, then remove finalizer
82
        if !res.DeletionTimestamp.IsZero() {
458✔
83
                log.V(1).Info("RenderTask is being deleted")
12✔
84
                r.Recorder.Eventf(res, nil, corev1.EventTypeWarning, "Deleting", "Delete", "RenderTask is being deleted, cleaning up secret and job")
12✔
85

12✔
86
                // Cleanup render resources, if exists
12✔
87
                if err := r.deleteRenderJob(ctx, res); err != nil && !apierrors.IsNotFound(err) {
12✔
88
                        return ctrlResult, errLogAndWrap(log, err, "failed to clean up render job")
×
89
                }
×
90

91
                if err := r.deleteConfigSecret(ctx, res); err != nil && !apierrors.IsNotFound(err) {
12✔
92
                        return ctrlResult, errLogAndWrap(log, err, "failed to clean up render secret")
×
93
                }
×
94

95
                if err := r.deleteAuthSecret(ctx, res); err != nil && !apierrors.IsNotFound(err) {
12✔
NEW
96
                        return ctrlResult, errLogAndWrap(log, err, "failed to clean up auth secret")
×
NEW
97
                }
×
98

99
                // Remove finalizer
100
                if slices.Contains(res.Finalizers, renderTaskFinalizer) {
24✔
101
                        log.V(1).Info("Removing finalizer from resource")
12✔
102
                        res.Finalizers = slices.DeleteFunc(res.Finalizers, func(f string) bool {
24✔
103
                                return f == renderTaskFinalizer
12✔
104
                        })
12✔
105
                        if err := r.Update(ctx, res); err != nil {
13✔
106
                                return ctrlResult, errLogAndWrap(log, err, "failed to remove finalizer")
1✔
107
                        }
1✔
108
                }
109

110
                return ctrlResult, nil
11✔
111
        }
112

113
        // Add finalizer if not present and not deleting
114
        if res.DeletionTimestamp.IsZero() {
868✔
115
                if !slices.Contains(res.Finalizers, renderTaskFinalizer) {
479✔
116
                        log.V(1).Info("Adding finalizer to resource")
45✔
117
                        res.Finalizers = append(res.Finalizers, renderTaskFinalizer)
45✔
118
                        if err := r.Update(ctx, res); err != nil {
45✔
119
                                return ctrlResult, errLogAndWrap(log, err, "failed to add finalizer")
×
120
                        }
×
121
                        // Return without requeue; the Update event will trigger reconciliation again
122
                        return ctrlResult, nil
45✔
123
                }
124
        }
125

126
        // Check if renderjob has already completed successfully
127
        sc := apimeta.FindStatusCondition(res.Status.Conditions, ConditionTypeJobSucceeded)
389✔
128
        if sc != nil && sc.ObservedGeneration >= res.Generation && sc.Status == metav1.ConditionTrue {
394✔
129
                log.V(1).Info("RenderTask has already completed successfully, no further action needed")
5✔
130
                return ctrlResult, nil
5✔
131
        }
5✔
132

133
        // Check if renderjob has already failed
134
        fc := apimeta.FindStatusCondition(res.Status.Conditions, ConditionTypeJobFailed)
384✔
135
        if fc != nil && fc.ObservedGeneration >= res.Generation && fc.Status == metav1.ConditionTrue {
387✔
136
                log.V(1).Info("RenderTask has already failed, no further action needed")
3✔
137
                return ctrlResult, nil
3✔
138
        }
3✔
139

140
        // Reconcile Config Secret
141
        configSecret := &corev1.Secret{}
381✔
142
        err := r.Get(ctx, r.configSecretKey(res), configSecret)
381✔
143
        if err != nil && apierrors.IsNotFound(err) {
473✔
144
                err := r.createConfigSecret(ctx, res)
92✔
145
                if err != nil {
143✔
146
                        r.Recorder.Eventf(res, nil, corev1.EventTypeWarning, "CreateSecretFailed", "CreateConfigSecret", fmt.Sprintf("Failed to create config secret: %s", err))
51✔
147
                        return ctrlResult, errLogAndWrap(log, err, "failed to create secret")
51✔
148
                }
51✔
149
        } else if err != nil {
289✔
150
                return ctrlResult, errLogAndWrap(log, err, "could not get secret")
×
151
        }
×
152

153
        // Reconcile Auth Secret
154
        authSecret := &corev1.Secret{}
330✔
155
        err = r.Get(ctx, r.authSecretKey(res), authSecret)
330✔
156
        if err != nil && apierrors.IsNotFound(err) && r.PushSecretRef != nil {
616✔
157
                err := r.copyAuthSecret(ctx, res)
286✔
158
                if err != nil {
559✔
159
                        r.Recorder.Eventf(res, nil, corev1.EventTypeWarning, "CreateSecretFailed", "CreateAuthSecret", fmt.Sprintf("Failed to create auth secret: %s", err))
273✔
160
                        return ctrlResult, errLogAndWrap(log, err, "failed to copy auth secret to namespace")
273✔
161
                }
273✔
162
        } else if client.IgnoreNotFound(err) != nil {
44✔
NEW
163
                return ctrlResult, errLogAndWrap(log, err, "could not get auth secret")
×
NEW
164
        }
×
165

166
        // Reconcile Job
167
        job := &batchv1.Job{}
57✔
168
        err = r.Get(ctx, r.renderJobKey(res), job)
57✔
169
        if err != nil && apierrors.IsNotFound(err) {
83✔
170
                err := r.createRenderJob(ctx, res, configSecret)
26✔
171
                if err != nil {
39✔
172
                        r.Recorder.Eventf(res, nil, corev1.EventTypeWarning, "CreateJobFailed", "CreateJob", fmt.Sprintf("Failed to create job: %s", err))
13✔
173
                        return ctrlResult, errLogAndWrap(log, err, "failed to create job")
13✔
174
                }
13✔
175
        } else if err != nil {
31✔
176
                return ctrlResult, errLogAndWrap(log, err, "could not get job")
×
177
        }
×
178

179
        // Update Status
180
        if changed := r.updateResourceStatusFromJob(ctx, res, job); changed {
61✔
181
                if err := r.Status().Update(ctx, res); err != nil {
17✔
182
                        return ctrlResult, errLogAndWrap(log, err, "failed to update status")
×
183
                }
×
184
        }
185

186
        // Check if we need to clean up
187
        if isJobComplete(job) && job.Status.Succeeded > 0 {
47✔
188
                if err := r.deleteRenderJob(ctx, res); err != nil && !apierrors.IsNotFound(err) {
3✔
189
                        r.Recorder.Eventf(res, job, corev1.EventTypeWarning, "DeletionFailed", "Delete", "Failed to delete job", err)
×
190
                        return ctrlResult, nil
×
191
                }
×
192
                if err := r.deleteConfigSecret(ctx, res); err != nil && !apierrors.IsNotFound(err) {
3✔
NEW
193
                        r.Recorder.Eventf(res, nil, corev1.EventTypeWarning, "DeletionFailed", "Delete", "Failed to delete config secret", err)
×
NEW
194
                        return ctrlResult, nil
×
NEW
195
                }
×
196
                if err := r.deleteAuthSecret(ctx, res); err != nil && !apierrors.IsNotFound(err) {
3✔
NEW
197
                        r.Recorder.Eventf(res, nil, corev1.EventTypeWarning, "DeletionFailed", "Delete", "Failed to delete auth secret", err)
×
198
                        return ctrlResult, nil
×
199
                }
×
200
                log.V(1).Info("Cleaned up after successful job")
3✔
201

3✔
202
                return ctrlResult, nil
3✔
203
        }
204

205
        // Check if job is still running
206
        if !isJobComplete(job) {
82✔
207
                log.V(1).Info("Job is still running, requeue after 5 seconds")
41✔
208
                return ctrl.Result{RequeueAfter: 5 * time.Second}, nil
41✔
209
        }
41✔
210

211
        return ctrlResult, nil
×
212
}
213

214
// updateResourceStatusFromJob updates the resource status based on job status
215
func (r *RenderTaskReconciler) updateResourceStatusFromJob(ctx context.Context, res *solarv1alpha1.RenderTask, job *batchv1.Job) (changed bool) {
44✔
216
        log := ctrl.LoggerFrom(ctx)
44✔
217

44✔
218
        if job == nil {
44✔
219
                changed = apimeta.SetStatusCondition(&res.Status.Conditions, metav1.Condition{
×
220
                        Type:               ConditionTypeJobScheduled,
×
221
                        Status:             metav1.ConditionFalse,
×
222
                        ObservedGeneration: res.Generation,
×
223
                        Reason:             "DoesNotExist",
×
224
                        Message:            "Renderer job does not exist",
×
225
                })
×
226

×
227
                return changed
×
228
        }
×
229

230
        if job.Status.Succeeded > 0 {
47✔
231
                changed = apimeta.SetStatusCondition(&res.Status.Conditions, metav1.Condition{
3✔
232
                        Type:               ConditionTypeJobSucceeded,
3✔
233
                        Status:             metav1.ConditionTrue,
3✔
234
                        ObservedGeneration: res.Generation,
3✔
235
                        Reason:             "JobSucceeded",
3✔
236
                        Message:            fmt.Sprintf("Renderer job completed successfully at %v", job.Status.CompletionTime),
3✔
237
                })
3✔
238

3✔
239
                if res.Status.ChartURL != r.referenceURL(res.Spec.Repository, res.Spec.Tag) {
6✔
240
                        res.Status.ChartURL = r.referenceURL(res.Spec.Repository, res.Spec.Tag)
3✔
241
                        changed = true
3✔
242
                }
3✔
243

244
                r.Recorder.Eventf(res, job, corev1.EventTypeNormal, "JobSucceeded", "RunJob", "Renderer job completed successfully")
3✔
245
                log.V(1).Info("Job succeeded", "name", job.Name)
3✔
246

3✔
247
                return changed
3✔
248
        }
249

250
        if job.Status.Failed > 0 {
42✔
251
                changed = apimeta.SetStatusCondition(&res.Status.Conditions, metav1.Condition{
1✔
252
                        Type:               ConditionTypeJobFailed,
1✔
253
                        Status:             metav1.ConditionTrue,
1✔
254
                        ObservedGeneration: res.Generation,
1✔
255
                        Reason:             "JobFailed",
1✔
256
                        Message:            "Renderer job failed",
1✔
257
                })
1✔
258
                r.Recorder.Eventf(res, job, corev1.EventTypeWarning, "JobFailed", "RunJob", "Renderer job failed")
1✔
259
                log.V(1).Info("Job failed", "name", job.Name)
1✔
260

1✔
261
                return changed
1✔
262
        }
1✔
263

264
        return apimeta.SetStatusCondition(&res.Status.Conditions, metav1.Condition{
40✔
265
                Type:               ConditionTypeJobScheduled,
40✔
266
                Status:             metav1.ConditionTrue,
40✔
267
                ObservedGeneration: res.Generation,
40✔
268
                Reason:             "JobScheduled",
40✔
269
                Message:            fmt.Sprintf("Renderer job is running (active: %d, succeeded: %d, failed: %d)", job.Status.Active, job.Status.Succeeded, job.Status.Failed),
40✔
270
        })
40✔
271
}
272

273
func (r *RenderTaskReconciler) deleteAuthSecret(ctx context.Context, res *solarv1alpha1.RenderTask) error {
15✔
274
        secret := &corev1.Secret{}
15✔
275
        if err := r.Get(ctx, r.authSecretKey(res), secret); err != nil {
26✔
276
                return err
11✔
277
        }
11✔
278

279
        return r.Delete(ctx, secret, client.PropagationPolicy(metav1.DeletePropagationBackground))
4✔
280
}
281

282
func (r *RenderTaskReconciler) deleteRenderJob(ctx context.Context, res *solarv1alpha1.RenderTask) error {
15✔
283
        job := &batchv1.Job{}
15✔
284
        if err := r.Get(ctx, r.renderJobKey(res), job); err != nil {
26✔
285
                return err
11✔
286
        }
11✔
287

288
        return r.Delete(ctx, job, client.PropagationPolicy(metav1.DeletePropagationBackground))
4✔
289
}
290

291
func (r *RenderTaskReconciler) deleteConfigSecret(ctx context.Context, res *solarv1alpha1.RenderTask) error {
15✔
292
        secret := &corev1.Secret{}
15✔
293
        if err := r.Get(ctx, r.configSecretKey(res), secret); err != nil {
16✔
294
                return err
1✔
295
        }
1✔
296

297
        return r.Delete(ctx, secret, client.PropagationPolicy(metav1.DeletePropagationBackground))
14✔
298
}
299

300
func (r *RenderTaskReconciler) copyAuthSecret(ctx context.Context, res *solarv1alpha1.RenderTask) error {
286✔
301
        log := ctrl.LoggerFrom(ctx)
286✔
302

286✔
303
        controllerSecret := &corev1.Secret{}
286✔
304
        if err := r.Get(ctx, client.ObjectKey{Name: r.PushSecretRef.Name, Namespace: r.PushSecretRef.Namespace}, controllerSecret); err != nil {
529✔
305
                return err
243✔
306
        }
243✔
307

308
        authSecret := &corev1.Secret{
43✔
309
                ObjectMeta: metav1.ObjectMeta{
43✔
310
                        Name:      r.authSecretKey(res).Name,
43✔
311
                        Namespace: r.authSecretKey(res).Namespace,
43✔
312
                },
43✔
313
                Type:       controllerSecret.Type,
43✔
314
                Data:       controllerSecret.Data,
43✔
315
                StringData: controllerSecret.StringData,
43✔
316
        }
43✔
317

43✔
318
        if err := r.Create(ctx, authSecret); err != nil {
73✔
319
                r.Recorder.Eventf(res, nil, corev1.EventTypeWarning, "CreationFailed", "Create", "Failed to create secret: %s", err)
30✔
320
                return errLogAndWrap(log, err, "secret creation failed")
30✔
321
        }
30✔
322

323
        // Set owner references
324
        if err := controllerutil.SetControllerReference(res, authSecret, r.Scheme); err != nil {
13✔
NEW
325
                return errLogAndWrap(log, err, "failed to set controller reference")
×
NEW
326
        }
×
327

328
        return nil
13✔
329
}
330

331
func (r *RenderTaskReconciler) createRenderJob(ctx context.Context, res *solarv1alpha1.RenderTask, secret *corev1.Secret) error {
26✔
332
        log := ctrl.LoggerFrom(ctx)
26✔
333

26✔
334
        jobName := r.renderJobKey(res).Name
26✔
335
        backoffLimit := int32(3)
26✔
336
        ttlSecondsAfterFinished := int32(3600) // Clean up after 1 hour
26✔
337

26✔
338
        job := &batchv1.Job{
26✔
339
                ObjectMeta: metav1.ObjectMeta{
26✔
340
                        Name:      jobName,
26✔
341
                        Namespace: r.renderJobKey(res).Namespace,
26✔
342
                        OwnerReferences: []metav1.OwnerReference{
26✔
343
                                {
26✔
344
                                        APIVersion:         solarv1alpha1.SchemeGroupVersion.String(),
26✔
345
                                        Kind:               res.Kind,
26✔
346
                                        Name:               res.Name,
26✔
347
                                        UID:                res.GetUID(),
26✔
348
                                        Controller:         ptr.To(true),
26✔
349
                                        BlockOwnerDeletion: ptr.To(true),
26✔
350
                                },
26✔
351
                        },
26✔
352
                        Annotations: map[string]string{
26✔
353
                                annotationJobName: jobName,
26✔
354
                        },
26✔
355
                },
26✔
356
                Spec: batchv1.JobSpec{
26✔
357
                        BackoffLimit:            &backoffLimit,
26✔
358
                        TTLSecondsAfterFinished: &ttlSecondsAfterFinished,
26✔
359
                        Template: corev1.PodTemplateSpec{
26✔
360
                                Spec: corev1.PodSpec{
26✔
361
                                        RestartPolicy: corev1.RestartPolicyNever,
26✔
362
                                        Containers: []corev1.Container{
26✔
363
                                                {
26✔
364
                                                        Name:    "renderer",
26✔
365
                                                        Image:   r.RendererImage,
26✔
366
                                                        Command: []string{r.RendererCommand},
26✔
367
                                                        Args: append(r.RendererArgs,
26✔
368
                                                                "/etc/renderer/config.json",
26✔
369
                                                                fmt.Sprintf(`--url="%s"`, r.referenceURL(res.Spec.Repository, res.Spec.Tag)),
26✔
370
                                                        ),
26✔
371
                                                        Env: []corev1.EnvVar{
26✔
372
                                                                {
26✔
373
                                                                        Name: "POD_NAMESPACE",
26✔
374
                                                                        ValueFrom: &corev1.EnvVarSource{
26✔
375
                                                                                FieldRef: &corev1.ObjectFieldSelector{
26✔
376
                                                                                        FieldPath: "metadata.namespace",
26✔
377
                                                                                },
26✔
378
                                                                        },
26✔
379
                                                                },
26✔
380
                                                                {
26✔
381
                                                                        Name: "POD_NAME",
26✔
382
                                                                        ValueFrom: &corev1.EnvVarSource{
26✔
383
                                                                                FieldRef: &corev1.ObjectFieldSelector{
26✔
384
                                                                                        FieldPath: "metadata.name",
26✔
385
                                                                                },
26✔
386
                                                                        },
26✔
387
                                                                },
26✔
388
                                                        },
26✔
389
                                                        VolumeMounts: []corev1.VolumeMount{
26✔
390
                                                                {
26✔
391
                                                                        Name:      "config",
26✔
392
                                                                        MountPath: "/etc/renderer/config.json",
26✔
393
                                                                        SubPath:   "config.json",
26✔
394
                                                                        ReadOnly:  true,
26✔
395
                                                                },
26✔
396
                                                        },
26✔
397
                                                },
26✔
398
                                        },
26✔
399
                                        Volumes: []corev1.Volume{
26✔
400
                                                {
26✔
401
                                                        Name: "config",
26✔
402
                                                        VolumeSource: corev1.VolumeSource{
26✔
403
                                                                Secret: &corev1.SecretVolumeSource{
26✔
404
                                                                        SecretName: secret.Name,
26✔
405
                                                                        Items: []corev1.KeyToPath{
26✔
406
                                                                                {
26✔
407
                                                                                        Key:  "config.json",
26✔
408
                                                                                        Path: "config.json",
26✔
409
                                                                                },
26✔
410
                                                                        },
26✔
411
                                                                },
26✔
412
                                                        },
26✔
413
                                                },
26✔
414
                                        },
26✔
415
                                },
26✔
416
                        },
26✔
417
                },
26✔
418
        }
26✔
419

26✔
420
        authSecret := &corev1.Secret{}
26✔
421
        if r.PushSecretRef != nil {
52✔
422
                if err := r.Get(ctx, r.authSecretKey(res), authSecret); err != nil {
26✔
NEW
423
                        if apierrors.IsNotFound(err) {
×
NEW
424
                                r.Recorder.Eventf(res, nil, corev1.EventTypeWarning, "SecretNotFound", "GetAuthCredentials", "the configured auth secret was not found")
×
NEW
425
                        }
×
426

NEW
427
                        return errLogAndWrap(log, err, "failed to get auth secret from secret ref")
×
428
                }
429
        }
430

431
        switch authSecret.Type {
26✔
432
        case corev1.SecretTypeBasicAuth:
2✔
433
                job.Spec.Template.Spec.Containers[0].EnvFrom = append(job.Spec.Template.Spec.Containers[0].EnvFrom, corev1.EnvFromSource{
2✔
434
                        SecretRef: &corev1.SecretEnvSource{
2✔
435
                                LocalObjectReference: corev1.LocalObjectReference{
2✔
436
                                        Name: authSecret.Name,
2✔
437
                                },
2✔
438
                        },
2✔
439
                })
2✔
440
                job.Spec.Template.Spec.Containers[0].Args = append(job.Spec.Template.Spec.Containers[0].Args,
2✔
441
                        `--username="$username"`, `--password="$password"`)
2✔
442

443
        case corev1.SecretTypeDockerConfigJson:
2✔
444
                job.Spec.Template.Spec.Volumes = append(job.Spec.Template.Spec.Volumes, corev1.Volume{
2✔
445
                        Name: "dockerconfig",
2✔
446
                        VolumeSource: corev1.VolumeSource{
2✔
447
                                Secret: &corev1.SecretVolumeSource{
2✔
448
                                        SecretName: authSecret.Name,
2✔
449
                                        Items: []corev1.KeyToPath{
2✔
450
                                                {
2✔
451
                                                        Key:  ".dockerconfigjson",
2✔
452
                                                        Path: "dockerconfig.json",
2✔
453
                                                },
2✔
454
                                        },
2✔
455
                                },
2✔
456
                        },
2✔
457
                })
2✔
458

2✔
459
                job.Spec.Template.Spec.Containers[0].VolumeMounts = append(job.Spec.Template.Spec.Containers[0].VolumeMounts, corev1.VolumeMount{
2✔
460
                        Name:      "dockerconfig",
2✔
461
                        MountPath: "/etc/renderer/dockerconfig.json",
2✔
462
                        SubPath:   "dockerconfig.json",
2✔
463
                        ReadOnly:  true,
2✔
464
                })
2✔
465

2✔
466
                job.Spec.Template.Spec.Containers[0].Env = append(job.Spec.Template.Spec.Containers[0].Env, corev1.EnvVar{
2✔
467
                        Name:  "DOCKER_CONFIG",
2✔
468
                        Value: "/etc/renderer/dockerconfig.json",
2✔
469
                })
2✔
470
        default:
22✔
471
        }
472

473
        if err := r.Create(ctx, job); err != nil {
39✔
474
                r.Recorder.Eventf(res, nil, corev1.EventTypeWarning, "CreationFailed", "Create", "Failed to create job: %s", err)
13✔
475
                return errLogAndWrap(log, err, "job creation failed")
13✔
476
        }
13✔
477

478
        // Set owner references
479
        if err := controllerutil.SetControllerReference(res, job, r.Scheme); err != nil {
13✔
480
                return errLogAndWrap(log, err, "failed to set controller reference")
×
481
        }
×
482

483
        res.Status.JobRef = &corev1.ObjectReference{
13✔
484
                APIVersion: batchv1.SchemeGroupVersion.String(),
13✔
485
                Kind:       "Job",
13✔
486
                Namespace:  job.Namespace,
13✔
487
                Name:       job.Name,
13✔
488
        }
13✔
489

13✔
490
        if err := r.Status().Update(ctx, res); err != nil {
13✔
491
                return errLogAndWrap(log, err, "failed to update status")
×
492
        }
×
493

494
        return nil
13✔
495
}
496

497
func (r *RenderTaskReconciler) createConfigSecret(ctx context.Context, res *solarv1alpha1.RenderTask) error {
92✔
498
        log := ctrl.LoggerFrom(ctx)
92✔
499

92✔
500
        cfgJson, err := json.Marshal(res.Spec.RendererConfig)
92✔
501
        if err != nil {
92✔
502
                return err
×
503
        }
×
504

505
        secret := &corev1.Secret{
92✔
506
                ObjectMeta: metav1.ObjectMeta{
92✔
507
                        Name:      r.configSecretKey(res).Name,
92✔
508
                        Namespace: r.configSecretKey(res).Namespace,
92✔
509
                        OwnerReferences: []metav1.OwnerReference{
92✔
510
                                {
92✔
511
                                        APIVersion:         solarv1alpha1.SchemeGroupVersion.String(),
92✔
512
                                        Kind:               res.Kind,
92✔
513
                                        Name:               res.Name,
92✔
514
                                        UID:                res.UID,
92✔
515
                                        Controller:         ptr.To(true),
92✔
516
                                        BlockOwnerDeletion: ptr.To(true),
92✔
517
                                },
92✔
518
                        },
92✔
519
                        Annotations: map[string]string{
92✔
520
                                annotationSecretName: r.configSecretKey(res).Name,
92✔
521
                        },
92✔
522
                },
92✔
523
                Type: corev1.SecretTypeOpaque,
92✔
524
                Data: map[string][]byte{
92✔
525
                        "config.json": cfgJson,
92✔
526
                },
92✔
527
        }
92✔
528

92✔
529
        if err := r.Create(ctx, secret); err != nil {
143✔
530
                r.Recorder.Eventf(res, nil, corev1.EventTypeWarning, "CreationFailed", "Create", "Failed to create secret: %s", err)
51✔
531
                return errLogAndWrap(log, err, "secret creation failed")
51✔
532
        }
51✔
533

534
        // Set owner references
535
        if err := controllerutil.SetControllerReference(res, secret, r.Scheme); err != nil {
41✔
536
                return errLogAndWrap(log, err, "failed to set controller reference")
×
537
        }
×
538

539
        res.Status.ConfigSecretRef = &corev1.ObjectReference{
41✔
540
                APIVersion: corev1.SchemeGroupVersion.String(),
41✔
541
                Kind:       "Secret",
41✔
542
                Namespace:  secret.Namespace,
41✔
543
                Name:       secret.Name,
41✔
544
        }
41✔
545

41✔
546
        if err := r.Status().Update(ctx, res); err != nil {
41✔
547
                return errLogAndWrap(log, err, "failed to update status")
×
548
        }
×
549

550
        return nil
41✔
551
}
552

553
func (r *RenderTaskReconciler) configSecretKey(res *solarv1alpha1.RenderTask) client.ObjectKey {
672✔
554
        return client.ObjectKey{
672✔
555
                Name:      fmt.Sprintf("render-%s", res.Name),
672✔
556
                Namespace: res.Namespace,
672✔
557
        }
672✔
558
}
672✔
559

560
func (r *RenderTaskReconciler) authSecretKey(res *solarv1alpha1.RenderTask) client.ObjectKey {
457✔
561
        return client.ObjectKey{
457✔
562
                Name:      fmt.Sprintf("auth-%s", res.Name),
457✔
563
                Namespace: res.Namespace,
457✔
564
        }
457✔
565
}
457✔
566

567
func (r *RenderTaskReconciler) renderJobKey(res *solarv1alpha1.RenderTask) client.ObjectKey {
124✔
568
        return client.ObjectKey{
124✔
569
                Name:      fmt.Sprintf("render-%s", res.Name),
124✔
570
                Namespace: res.Namespace,
124✔
571
        }
124✔
572
}
124✔
573

574
// isJobComplete returns true if the Job is complete
575
func isJobComplete(job *batchv1.Job) bool {
85✔
576
        return job.Status.CompletionTime != nil
85✔
577
}
85✔
578

579
func (r *RenderTaskReconciler) referenceURL(repo string, tag string) string {
32✔
580
        base := r.BaseURL
32✔
581
        if !strings.HasPrefix(base, "oci://") {
64✔
582
                base = fmt.Sprintf("oci://%s", base)
32✔
583
        }
32✔
584
        base = strings.TrimSuffix(base, "/")
32✔
585
        url := fmt.Sprintf("%s/%s:%s", base, repo, tag)
32✔
586

32✔
587
        return url
32✔
588
}
589

590
// SetupWithManager sets up the controller with the Manager.
591
func (r *RenderTaskReconciler) SetupWithManager(mgr ctrl.Manager) error {
1✔
592
        return ctrl.NewControllerManagedBy(mgr).
1✔
593
                For(&solarv1alpha1.RenderTask{}).
1✔
594
                Owns(&batchv1.Job{}).
1✔
595
                Owns(&corev1.Secret{}).
1✔
596
                Complete(r)
1✔
597
}
1✔
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