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

opendefensecloud / solution-arsenal / 21944505903

12 Feb 2026 11:20AM UTC coverage: 72.872% (+5.4%) from 67.462%
21944505903

Pull #147

github

web-flow
Merge 8757da111 into 717f548f7
Pull Request #147: Implement RenderTask controller

680 of 791 new or added lines in 10 files covered. (85.97%)

41 existing lines in 3 files now uncovered.

1515 of 2079 relevant lines covered (72.87%)

15.88 hits per line

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

87.28
/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
        "time"
12

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

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

27
const (
28
        renderTaskFinalizer = "solar.opendefense.cloud/rendertask-finalizer"
29

30
        annotationJobName    = "solar.opendefense.cloud/job-name"
31
        annotationSecretName = "solar.opendefense.cloud/secret-name"
32

33
        // Condition types
34
        ConditionTypeJobScheduled = "JobScheduled"
35
        ConditionTypeJobSucceeded = "JobSucceeded"
36
        ConditionTypeJobFailed    = "JobFailed"
37
        ConditionTypeSecretSynced = "SecretSynced"
38

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

43
// RenderTaskReconciler reconciles a RenderTask object
44
type RenderTaskReconciler struct {
45
        client.Client
46
        Scheme          *runtime.Scheme
47
        Recorder        record.EventRecorder
48
        RendererImage   string
49
        RendererCommand string
50
        RendererArgs    []string
51
}
52

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

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

192✔
65
        log.V(1).Info("RenderTask is being reconciled", "req", req)
192✔
66

192✔
67
        // Fetch the RenderTask instance
192✔
68
        res := &solarv1alpha1.RenderTask{}
192✔
69
        if err := r.Get(ctx, req.NamespacedName, res); err != nil {
198✔
70
                if apierrors.IsNotFound(err) {
12✔
71
                        // Object not found, return. Created objects are automatically garbage collected.
6✔
72
                        return ctrlResult, nil
6✔
73
                }
6✔
74

NEW
75
                return ctrlResult, errLogAndWrap(log, err, "failed to get object")
×
76
        }
77

78
        // Handle deletion: cleanup job and secret, then remove finalizer
79
        if !res.DeletionTimestamp.IsZero() {
192✔
80
                log.V(1).Info("RenderTask is being deleted")
6✔
81
                r.Recorder.Event(res, corev1.EventTypeWarning, "Deleting", "RenderTask is being deleted, cleaning up secret and job")
6✔
82

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

88
                if err := r.deleteRenderSecret(ctx, res); err != nil && !apierrors.IsNotFound(err) {
6✔
NEW
89
                        return ctrlResult, errLogAndWrap(log, err, "failed to clean up render job")
×
NEW
90
                }
×
91

92
                // Remove finalizer
93
                if slices.Contains(res.Finalizers, renderTaskFinalizer) {
12✔
94
                        log.V(1).Info("Removing finalizer from resource")
6✔
95
                        res.Finalizers = slices.DeleteFunc(res.Finalizers, func(f string) bool {
12✔
96
                                return f == renderTaskFinalizer
6✔
97
                        })
6✔
98
                        if err := r.Update(ctx, res); err != nil {
7✔
99
                                return ctrlResult, errLogAndWrap(log, err, "failed to remove finalizer")
1✔
100
                        }
1✔
101
                }
102

103
                return ctrlResult, nil
5✔
104
        }
105

106
        // Add finalizer if not present and not deleting
107
        if res.DeletionTimestamp.IsZero() {
360✔
108
                if !slices.Contains(res.Finalizers, renderTaskFinalizer) {
209✔
109
                        log.V(1).Info("Adding finalizer to resource")
29✔
110
                        res.Finalizers = append(res.Finalizers, renderTaskFinalizer)
29✔
111
                        if err := r.Update(ctx, res); err != nil {
29✔
NEW
112
                                return ctrlResult, errLogAndWrap(log, err, "failed to add finalizer")
×
NEW
113
                        }
×
114
                        // Return without requeue; the Update event will trigger reconciliation again
115
                        return ctrlResult, nil
29✔
116
                }
117
        }
118

119
        // Check if renderjob has already completed successfully
120
        sc := apimeta.FindStatusCondition(res.Status.Conditions, ConditionTypeJobSucceeded)
151✔
121
        if sc != nil && sc.ObservedGeneration >= res.Generation && sc.Status == metav1.ConditionTrue {
157✔
122
                log.V(1).Info("RenderTask has already completed successfully, no further action needed")
6✔
123
                return ctrlResult, nil
6✔
124
        }
6✔
125

126
        // Check if renderjob has already failed
127
        fc := apimeta.FindStatusCondition(res.Status.Conditions, ConditionTypeJobFailed)
145✔
128
        if fc != nil && fc.ObservedGeneration >= res.Generation && fc.Status == metav1.ConditionTrue {
148✔
129
                log.V(1).Info("RenderTask has already failed, no further action needed")
3✔
130
                return ctrlResult, nil
3✔
131
        }
3✔
132

133
        // Check if secret creation has already failed
134
        ssc := apimeta.FindStatusCondition(res.Status.Conditions, ConditionTypeSecretSynced)
142✔
135
        if ssc != nil && ssc.ObservedGeneration >= res.Generation && ssc.Status == metav1.ConditionFalse {
145✔
136
                log.V(1).Info("Secret creation failed, aborting reconcile")
3✔
137
                return ctrlResult, nil
3✔
138
        }
3✔
139

140
        // Reconcile Secret
141
        secret := &corev1.Secret{}
139✔
142
        err := r.Get(ctx, client.ObjectKey{Name: renderPrefixed(res.Name), Namespace: res.Namespace}, secret)
139✔
143
        if err != nil && apierrors.IsNotFound(err) {
168✔
144
                err := r.createRenderSecret(ctx, res)
29✔
145
                if err != nil {
32✔
146
                        r.Recorder.Event(res, corev1.EventTypeWarning, "CreateSecretFailed", fmt.Sprintf("Failed to create secret: %s", err))
3✔
147
                        if changed := apimeta.SetStatusCondition(&res.Status.Conditions, metav1.Condition{
3✔
148
                                Type:               ConditionTypeSecretSynced,
3✔
149
                                Status:             metav1.ConditionFalse,
3✔
150
                                ObservedGeneration: res.Generation,
3✔
151
                                Reason:             "SecretCreationFailed",
3✔
152
                                Message:            fmt.Sprintf("Failed to create secret: %v", err),
3✔
153
                        }); changed {
6✔
154
                                if err := r.Status().Update(ctx, res); err != nil {
3✔
NEW
155
                                        log.Error(err, "failed to update RenderTask status")
×
NEW
156
                                }
×
157
                        }
158

159
                        return ctrlResult, errLogAndWrap(log, err, "failed to create secret")
3✔
160
                }
161
                // update secret after creation
162
                if err := r.Get(ctx, client.ObjectKey{Name: renderPrefixed(res.Name), Namespace: res.Namespace}, secret); err != nil {
26✔
NEW
163
                        return ctrlResult, errLogAndWrap(log, err, "could not get secret")
×
NEW
164
                }
×
165
        } else if err != nil {
110✔
NEW
166
                return ctrlResult, errLogAndWrap(log, err, "could not get secret")
×
NEW
167
        }
×
168

169
        // Reconcile Job
170
        job := &batchv1.Job{}
136✔
171
        err = r.Get(ctx, client.ObjectKey{Name: renderPrefixed(res.Name), Namespace: res.Namespace}, job)
136✔
172
        if err != nil && apierrors.IsNotFound(err) {
180✔
173
                err := r.createRenderJob(ctx, res, secret)
44✔
174
                if err != nil {
64✔
175
                        r.Recorder.Event(res, corev1.EventTypeWarning, "CreateJobFailed", fmt.Sprintf("Failed to create job: %s", err))
20✔
176
                        return ctrlResult, errLogAndWrap(log, err, "failed to create job")
20✔
177
                }
20✔
178

179
                // update job after creation
180
                if err := r.Get(ctx, client.ObjectKey{Name: renderPrefixed(res.Name), Namespace: res.Namespace}, job); err != nil {
24✔
NEW
181
                        return ctrlResult, errLogAndWrap(log, err, "could not get job")
×
NEW
182
                }
×
183
        } else if err != nil {
92✔
NEW
184
                return ctrlResult, errLogAndWrap(log, err, "could not get job")
×
NEW
185
        }
×
186

187
        if changed := r.updateResourceStatusFromJob(ctx, res, job); changed {
150✔
188
                if err := r.Status().Update(ctx, res); err != nil {
40✔
189
                        return ctrlResult, errLogAndWrap(log, err, "failed to update status")
6✔
190
                }
6✔
191
        }
192

193
        // Check if we need to clean up
194
        if isJobComplete(job) && job.Status.Succeeded > 0 {
113✔
195
                if err := r.deleteRenderJob(ctx, res); err != nil && !apierrors.IsNotFound(err) {
3✔
NEW
196
                        r.Recorder.Eventf(res, corev1.EventTypeWarning, "DeletionFailed", "Failed to delete job", err)
×
NEW
197
                        return ctrlResult, nil
×
NEW
198
                }
×
199
                if err := r.deleteRenderSecret(ctx, res); err != nil && !apierrors.IsNotFound(err) {
3✔
NEW
200
                        r.Recorder.Eventf(res, corev1.EventTypeWarning, "DeletionFailed", "Failed to delete secret", err)
×
NEW
201
                        return ctrlResult, nil
×
NEW
202
                }
×
203
                log.V(1).Info("Cleaned up after successful job")
3✔
204

3✔
205
                return ctrlResult, nil
3✔
206
        }
207

208
        // Check if job is still running
209
        if !isJobComplete(job) {
214✔
210
                log.V(1).Info("Job is still running, requeue after 5 seconds")
107✔
211
                return ctrl.Result{RequeueAfter: 5 * time.Second}, nil
107✔
212
        }
107✔
213

NEW
214
        return ctrlResult, nil
×
215
}
216

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

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

×
NEW
230
                return changed
×
NEW
231
        }
×
232

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

3✔
242
                if res.Status.ChartURL != res.Spec.PushOptions.ReferenceURL {
6✔
243
                        res.Status.ChartURL = res.Spec.PushOptions.ReferenceURL
3✔
244
                        changed = true
3✔
245
                }
3✔
246

247
                r.Recorder.Event(res, corev1.EventTypeNormal, "JobSucceeded", "Renderer job completed successfully")
3✔
248
                log.V(1).Info("Job succeeded", "name", job.Name)
3✔
249

3✔
250
                return changed
3✔
251
        }
252

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

1✔
264
                return changed
1✔
265
        }
1✔
266

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

276
func (r *RenderTaskReconciler) deleteRenderJob(ctx context.Context, res *solarv1alpha1.RenderTask) error {
9✔
277
        job := &batchv1.Job{}
9✔
278
        if err := r.Get(ctx, client.ObjectKey{Name: renderPrefixed(res.Name), Namespace: res.Namespace}, job); err != nil {
10✔
279
                return err
1✔
280
        }
1✔
281

282
        return r.Delete(ctx, job, client.PropagationPolicy(metav1.DeletePropagationBackground))
8✔
283
}
284

285
func (r *RenderTaskReconciler) deleteRenderSecret(ctx context.Context, res *solarv1alpha1.RenderTask) error {
9✔
286
        secret := &corev1.Secret{}
9✔
287
        if err := r.Get(ctx, client.ObjectKey{Name: renderPrefixed(res.Name), Namespace: res.Namespace}, secret); err != nil {
10✔
288
                return err
1✔
289
        }
1✔
290

291
        return r.Delete(ctx, secret, client.PropagationPolicy(metav1.DeletePropagationBackground))
8✔
292
}
293

294
func (r *RenderTaskReconciler) createRenderJob(ctx context.Context, res *solarv1alpha1.RenderTask, secret *corev1.Secret) error {
44✔
295
        log := ctrl.LoggerFrom(ctx)
44✔
296

44✔
297
        jobName := renderPrefixed(res.Name)
44✔
298
        backoffLimit := int32(3)
44✔
299
        ttlSecondsAfterFinished := int32(3600) // Clean up after 1 hour
44✔
300

44✔
301
        job := &batchv1.Job{
44✔
302
                ObjectMeta: metav1.ObjectMeta{
44✔
303
                        Name:      jobName,
44✔
304
                        Namespace: res.Namespace,
44✔
305
                        OwnerReferences: []metav1.OwnerReference{
44✔
306
                                {
44✔
307
                                        APIVersion:         solarv1alpha1.SchemeGroupVersion.String(),
44✔
308
                                        Kind:               res.Kind,
44✔
309
                                        Name:               res.Name,
44✔
310
                                        UID:                res.GetUID(),
44✔
311
                                        Controller:         boolPtr(true),
44✔
312
                                        BlockOwnerDeletion: boolPtr(true),
44✔
313
                                },
44✔
314
                        },
44✔
315
                        Annotations: map[string]string{
44✔
316
                                annotationJobName: jobName,
44✔
317
                        },
44✔
318
                },
44✔
319
                Spec: batchv1.JobSpec{
44✔
320
                        BackoffLimit:            &backoffLimit,
44✔
321
                        TTLSecondsAfterFinished: &ttlSecondsAfterFinished,
44✔
322
                        Template: corev1.PodTemplateSpec{
44✔
323
                                Spec: corev1.PodSpec{
44✔
324
                                        RestartPolicy: corev1.RestartPolicyNever,
44✔
325
                                        Containers: []corev1.Container{
44✔
326
                                                {
44✔
327
                                                        Name:    "renderer",
44✔
328
                                                        Image:   r.RendererImage,
44✔
329
                                                        Command: []string{r.RendererCommand},
44✔
330
                                                        Args:    append(r.RendererArgs, "/etc/renderer/config.json"),
44✔
331
                                                        Env: []corev1.EnvVar{
44✔
332
                                                                {
44✔
333
                                                                        Name: "POD_NAMESPACE",
44✔
334
                                                                        ValueFrom: &corev1.EnvVarSource{
44✔
335
                                                                                FieldRef: &corev1.ObjectFieldSelector{
44✔
336
                                                                                        FieldPath: "metadata.namespace",
44✔
337
                                                                                },
44✔
338
                                                                        },
44✔
339
                                                                },
44✔
340
                                                                {
44✔
341
                                                                        Name: "POD_NAME",
44✔
342
                                                                        ValueFrom: &corev1.EnvVarSource{
44✔
343
                                                                                FieldRef: &corev1.ObjectFieldSelector{
44✔
344
                                                                                        FieldPath: "metadata.name",
44✔
345
                                                                                },
44✔
346
                                                                        },
44✔
347
                                                                },
44✔
348
                                                        },
44✔
349
                                                        VolumeMounts: []corev1.VolumeMount{
44✔
350
                                                                {
44✔
351
                                                                        Name:      "config",
44✔
352
                                                                        MountPath: "/etc/renderer",
44✔
353
                                                                        ReadOnly:  true,
44✔
354
                                                                },
44✔
355
                                                        },
44✔
356
                                                },
44✔
357
                                        },
44✔
358
                                        Volumes: []corev1.Volume{
44✔
359
                                                {
44✔
360
                                                        Name: "config",
44✔
361
                                                        VolumeSource: corev1.VolumeSource{
44✔
362
                                                                Secret: &corev1.SecretVolumeSource{
44✔
363
                                                                        SecretName: secret.Name,
44✔
364
                                                                        Items: []corev1.KeyToPath{
44✔
365
                                                                                {
44✔
366
                                                                                        Key:  "config.json",
44✔
367
                                                                                        Path: "config.json",
44✔
368
                                                                                },
44✔
369
                                                                        },
44✔
370
                                                                },
44✔
371
                                                        },
44✔
372
                                                },
44✔
373
                                        },
44✔
374
                                },
44✔
375
                        },
44✔
376
                },
44✔
377
        }
44✔
378

44✔
379
        if err := r.Create(ctx, job); err != nil {
64✔
380
                r.Recorder.Eventf(res, corev1.EventTypeWarning, "CreationFailed", "Failed to create job: %s", err)
20✔
381
                return errLogAndWrap(log, err, "job creation failed")
20✔
382
        }
20✔
383

384
        // Set owner references
385
        if err := controllerutil.SetControllerReference(res, job, r.Scheme); err != nil {
24✔
NEW
386
                return errLogAndWrap(log, err, "failed to set controller reference")
×
NEW
387
        }
×
388

389
        res.Status.JobRef = &corev1.ObjectReference{
24✔
390
                APIVersion: batchv1.SchemeGroupVersion.String(),
24✔
391
                Kind:       "Job",
24✔
392
                Namespace:  job.Namespace,
24✔
393
                Name:       job.Name,
24✔
394
        }
24✔
395

24✔
396
        if err := r.Status().Update(ctx, res); err != nil {
24✔
NEW
397
                return errLogAndWrap(log, err, "failed to update status")
×
NEW
398
        }
×
399

400
        return nil
24✔
401
}
402

403
func (r *RenderTaskReconciler) createRenderSecret(ctx context.Context, res *solarv1alpha1.RenderTask) error {
29✔
404
        log := ctrl.LoggerFrom(ctx)
29✔
405

29✔
406
        cfgJson, err := json.Marshal(res.Spec.RendererConfig)
29✔
407
        if err != nil {
29✔
NEW
408
                return err
×
NEW
409
        }
×
410

411
        secret := &corev1.Secret{
29✔
412
                ObjectMeta: metav1.ObjectMeta{
29✔
413
                        Name:      renderPrefixed(res.Name),
29✔
414
                        Namespace: res.Namespace,
29✔
415
                        OwnerReferences: []metav1.OwnerReference{
29✔
416
                                {
29✔
417
                                        APIVersion:         solarv1alpha1.SchemeGroupVersion.String(),
29✔
418
                                        Kind:               res.Kind,
29✔
419
                                        Name:               res.Name,
29✔
420
                                        UID:                res.UID,
29✔
421
                                        Controller:         boolPtr(true),
29✔
422
                                        BlockOwnerDeletion: boolPtr(true),
29✔
423
                                },
29✔
424
                        },
29✔
425
                        Annotations: map[string]string{
29✔
426
                                annotationSecretName: renderPrefixed(res.Name),
29✔
427
                        },
29✔
428
                },
29✔
429
                Type: corev1.SecretTypeOpaque,
29✔
430
                Data: map[string][]byte{
29✔
431
                        "config.json": cfgJson,
29✔
432
                },
29✔
433
        }
29✔
434

29✔
435
        if err := r.Create(ctx, secret); err != nil {
32✔
436
                r.Recorder.Eventf(res, corev1.EventTypeWarning, "CreationFailed", "Failed to create secret: %s", err)
3✔
437
                return errLogAndWrap(log, err, "secret creation failed")
3✔
438
        }
3✔
439

440
        // Set owner references
441
        if err := controllerutil.SetControllerReference(res, secret, r.Scheme); err != nil {
26✔
NEW
442
                return errLogAndWrap(log, err, "failed to set controller reference")
×
NEW
443
        }
×
444

445
        res.Status.ConfigSecretRef = &corev1.ObjectReference{
26✔
446
                APIVersion: corev1.SchemeGroupVersion.String(),
26✔
447
                Kind:       "Secret",
26✔
448
                Namespace:  secret.Namespace,
26✔
449
                Name:       secret.Name,
26✔
450
        }
26✔
451

26✔
452
        if err := r.Status().Update(ctx, res); err != nil {
26✔
NEW
453
                return errLogAndWrap(log, err, "failed to update status")
×
NEW
454
        }
×
455

456
        return nil
26✔
457
}
458

459
// renderPrefixed returns the "render-" prefixed string
460
func renderPrefixed(r string) string {
445✔
461
        return fmt.Sprintf("render-%s", r)
445✔
462
}
445✔
463

464
// isJobComplete returns true if the Job is complete
465
func isJobComplete(job *batchv1.Job) bool {
217✔
466
        return job.Status.CompletionTime != nil
217✔
467
}
217✔
468

469
func boolPtr(b bool) *bool {
146✔
470
        return &b
146✔
471
}
146✔
472

473
// SetupWithManager sets up the controller with the Manager.
474
func (r *RenderTaskReconciler) SetupWithManager(mgr ctrl.Manager) error {
1✔
475
        return ctrl.NewControllerManagedBy(mgr).
1✔
476
                For(&solarv1alpha1.RenderTask{}).
1✔
477
                Owns(&batchv1.Job{}).
1✔
478
                Owns(&corev1.Secret{}).
1✔
479
                Complete(r)
1✔
480
}
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