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

opendefensecloud / solution-arsenal / 22763955253

06 Mar 2026 12:43PM UTC coverage: 70.752% (+0.008%) from 70.744%
22763955253

push

github

web-flow
Reference Credentials from Secret (#195)

RenderTask controller has a single secret reference that gets copied to the destination namespace in order to get mounted.

240 of 261 new or added lines in 5 files covered. (91.95%)

240 existing lines in 13 files now uncovered.

2003 of 2831 relevant lines covered (70.75%)

23.72 hits per line

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

89.46
/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) {
469✔
65
        log := ctrl.LoggerFrom(ctx)
469✔
66
        ctrlResult := ctrl.Result{}
469✔
67

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

469✔
70
        // Fetch the RenderTask instance
469✔
71
        res := &solarv1alpha1.RenderTask{}
469✔
72
        if err := r.Get(ctx, req.NamespacedName, res); err != nil {
481✔
73
                if apierrors.IsNotFound(err) {
24✔
74
                        // Object not found, return. Created objects are automatically garbage collected.
12✔
75
                        return ctrlResult, nil
12✔
76
                }
12✔
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() {
472✔
83
                log.V(1).Info("RenderTask is being deleted")
15✔
84
                r.Recorder.Eventf(res, nil, corev1.EventTypeWarning, "Deleting", "Delete", "RenderTask is being deleted, cleaning up secret and job")
15✔
85

15✔
86
                // Cleanup render resources, if exists
15✔
87
                if err := r.deleteRenderJob(ctx, res); err != nil && !apierrors.IsNotFound(err) {
15✔
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) {
15✔
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) {
15✔
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) {
30✔
101
                        log.V(1).Info("Removing finalizer from resource")
15✔
102
                        res.Finalizers = slices.DeleteFunc(res.Finalizers, func(f string) bool {
30✔
103
                                return f == renderTaskFinalizer
15✔
104
                        })
15✔
105
                        if err := r.Update(ctx, res); err != nil {
18✔
106
                                return ctrlResult, errLogAndWrap(log, err, "failed to remove finalizer")
3✔
107
                        }
3✔
108
                }
109

110
                return ctrlResult, nil
12✔
111
        }
112

113
        // Add finalizer if not present and not deleting
114
        if res.DeletionTimestamp.IsZero() {
884✔
115
                if !slices.Contains(res.Finalizers, renderTaskFinalizer) {
488✔
116
                        log.V(1).Info("Adding finalizer to resource")
46✔
117
                        res.Finalizers = append(res.Finalizers, renderTaskFinalizer)
46✔
118
                        if err := r.Update(ctx, res); err != nil {
46✔
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
46✔
123
                }
124
        }
125

126
        // Check if renderjob has already completed successfully
127
        sc := apimeta.FindStatusCondition(res.Status.Conditions, ConditionTypeJobSucceeded)
396✔
128
        if sc != nil && sc.ObservedGeneration >= res.Generation && sc.Status == metav1.ConditionTrue {
401✔
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)
391✔
135
        if fc != nil && fc.ObservedGeneration >= res.Generation && fc.Status == metav1.ConditionTrue {
394✔
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{}
388✔
142
        err := r.Get(ctx, r.configSecretKey(res), configSecret)
388✔
143
        if err != nil && apierrors.IsNotFound(err) {
476✔
144
                err := r.createConfigSecret(ctx, res)
88✔
145
                if err != nil {
134✔
146
                        r.Recorder.Eventf(res, nil, corev1.EventTypeWarning, "CreateSecretFailed", "CreateConfigSecret", fmt.Sprintf("Failed to create config secret: %s", err))
46✔
147
                        return ctrlResult, errLogAndWrap(log, err, "failed to create secret")
46✔
148
                }
46✔
149
        } else if err != nil {
300✔
UNCOV
150
                return ctrlResult, errLogAndWrap(log, err, "could not get secret")
×
151
        }
×
152

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

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

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

186
        // Check if we need to clean up
187
        if isJobComplete(job) && job.Status.Succeeded > 0 {
132✔
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) {
252✔
207
                log.V(1).Info("Job is still running, requeue after 5 seconds")
126✔
208
                return ctrl.Result{RequeueAfter: 5 * time.Second}, nil
126✔
209
        }
126✔
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) {
136✔
216
        log := ctrl.LoggerFrom(ctx)
136✔
217

136✔
218
        if job == nil {
136✔
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 {
139✔
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 {
134✔
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{
132✔
265
                Type:               ConditionTypeJobScheduled,
132✔
266
                Status:             metav1.ConditionTrue,
132✔
267
                ObservedGeneration: res.Generation,
132✔
268
                Reason:             "JobScheduled",
132✔
269
                Message:            fmt.Sprintf("Renderer job is running (active: %d, succeeded: %d, failed: %d)", job.Status.Active, job.Status.Succeeded, job.Status.Failed),
132✔
270
        })
132✔
271
}
272

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

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

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

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

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

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

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

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

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

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

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

328
        return nil
26✔
329
}
330

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

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

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

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

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

431
        switch authSecret.Type {
43✔
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:
23✔
444
                job.Spec.Template.Spec.Volumes = append(job.Spec.Template.Spec.Volumes, corev1.Volume{
23✔
445
                        Name: "dockerconfig",
23✔
446
                        VolumeSource: corev1.VolumeSource{
23✔
447
                                Secret: &corev1.SecretVolumeSource{
23✔
448
                                        SecretName: authSecret.Name,
23✔
449
                                        Items: []corev1.KeyToPath{
23✔
450
                                                {
23✔
451
                                                        Key:  ".dockerconfigjson",
23✔
452
                                                        Path: "dockerconfig.json",
23✔
453
                                                },
23✔
454
                                        },
23✔
455
                                },
23✔
456
                        },
23✔
457
                })
23✔
458

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

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

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

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

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

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

494
        return nil
26✔
495
}
496

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

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

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

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

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

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

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

550
        return nil
42✔
551
}
552

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

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

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

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

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

58✔
587
        return url
58✔
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