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

opendefensecloud / solution-arsenal / 21944812532

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

Pull #147

github

web-flow
Merge 8638f106e 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.95 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) {
180✔
62
        log := ctrl.LoggerFrom(ctx)
180✔
63
        ctrlResult := ctrl.Result{}
180✔
64

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

180✔
67
        // Fetch the RenderTask instance
180✔
68
        res := &solarv1alpha1.RenderTask{}
180✔
69
        if err := r.Get(ctx, req.NamespacedName, res); err != nil {
186✔
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() {
181✔
80
                log.V(1).Info("RenderTask is being deleted")
7✔
81
                r.Recorder.Event(res, corev1.EventTypeWarning, "Deleting", "RenderTask is being deleted, cleaning up secret and job")
7✔
82

7✔
83
                // Cleanup render resources, if exists
7✔
84
                if err := r.deleteRenderJob(ctx, res); err != nil && !apierrors.IsNotFound(err) {
7✔
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) {
7✔
NEW
89
                        return ctrlResult, errLogAndWrap(log, err, "failed to clean up render secret")
×
NEW
90
                }
×
91

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

103
                return ctrlResult, nil
5✔
104
        }
105

106
        // Add finalizer if not present and not deleting
107
        if res.DeletionTimestamp.IsZero() {
334✔
108
                if !slices.Contains(res.Finalizers, renderTaskFinalizer) {
196✔
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)
138✔
121
        if sc != nil && sc.ObservedGeneration >= res.Generation && sc.Status == metav1.ConditionTrue {
143✔
122
                log.V(1).Info("RenderTask has already completed successfully, no further action needed")
5✔
123
                return ctrlResult, nil
5✔
124
        }
5✔
125

126
        // Check if renderjob has already failed
127
        fc := apimeta.FindStatusCondition(res.Status.Conditions, ConditionTypeJobFailed)
133✔
128
        if fc != nil && fc.ObservedGeneration >= res.Generation && fc.Status == metav1.ConditionTrue {
136✔
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)
130✔
135
        if ssc != nil && ssc.ObservedGeneration >= res.Generation && ssc.Status == metav1.ConditionFalse {
134✔
136
                log.V(1).Info("Secret creation failed, aborting reconcile")
4✔
137
                return ctrlResult, nil
4✔
138
        }
4✔
139

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

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

169
        // Reconcile Job
170
        job := &batchv1.Job{}
122✔
171
        err = r.Get(ctx, client.ObjectKey{Name: renderPrefixed(res.Name), Namespace: res.Namespace}, job)
122✔
172
        if err != nil && apierrors.IsNotFound(err) {
165✔
173
                err := r.createRenderJob(ctx, res, secret)
43✔
174
                if err != nil {
63✔
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 {
23✔
NEW
181
                        return ctrlResult, errLogAndWrap(log, err, "could not get job")
×
NEW
182
                }
×
183
        } else if err != nil {
79✔
NEW
184
                return ctrlResult, errLogAndWrap(log, err, "could not get job")
×
NEW
185
        }
×
186

187
        if changed := r.updateResourceStatusFromJob(ctx, res, job); changed {
135✔
188
                if err := r.Status().Update(ctx, res); err != nil {
39✔
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 {
99✔
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) {
186✔
210
                log.V(1).Info("Job is still running, requeue after 5 seconds")
93✔
211
                return ctrl.Result{RequeueAfter: 5 * time.Second}, nil
93✔
212
        }
93✔
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) {
102✔
219
        log := ctrl.LoggerFrom(ctx)
102✔
220

102✔
221
        if job == nil {
102✔
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 {
105✔
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 {
100✔
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{
98✔
268
                Type:               ConditionTypeJobScheduled,
98✔
269
                Status:             metav1.ConditionTrue,
98✔
270
                ObservedGeneration: res.Generation,
98✔
271
                Reason:             "JobScheduled",
98✔
272
                Message:            fmt.Sprintf("Renderer job is running (active: %d, succeeded: %d, failed: %d)", job.Status.Active, job.Status.Succeeded, job.Status.Failed),
98✔
273
        })
98✔
274
}
275

276
func (r *RenderTaskReconciler) deleteRenderJob(ctx context.Context, res *solarv1alpha1.RenderTask) error {
10✔
277
        job := &batchv1.Job{}
10✔
278
        if err := r.Get(ctx, client.ObjectKey{Name: renderPrefixed(res.Name), Namespace: res.Namespace}, job); err != nil {
12✔
279
                return err
2✔
280
        }
2✔
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 {
10✔
286
        secret := &corev1.Secret{}
10✔
287
        if err := r.Get(ctx, client.ObjectKey{Name: renderPrefixed(res.Name), Namespace: res.Namespace}, secret); err != nil {
12✔
288
                return err
2✔
289
        }
2✔
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 {
43✔
295
        log := ctrl.LoggerFrom(ctx)
43✔
296

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

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

43✔
379
        if err := r.Create(ctx, job); err != nil {
63✔
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 {
23✔
NEW
386
                return errLogAndWrap(log, err, "failed to set controller reference")
×
NEW
387
        }
×
388

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

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

400
        return nil
23✔
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 {
33✔
436
                r.Recorder.Eventf(res, corev1.EventTypeWarning, "CreationFailed", "Failed to create secret: %s", err)
4✔
437
                return errLogAndWrap(log, err, "secret creation failed")
4✔
438
        }
4✔
439

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

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

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

456
        return nil
25✔
457
}
458

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

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

469
func boolPtr(b bool) *bool {
144✔
470
        return &b
144✔
471
}
144✔
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