Skip to content

Commit

Permalink
fix: use correct path to PodSpec for CronJob
Browse files Browse the repository at this point in the history
CronJob has a different path to the PodSpec from the other currently
supported types of ImageRegistryTransform. This fixes the image
transform logic to find the correct path.
  • Loading branch information
sdowell committed May 23, 2024
1 parent 83bd9c0 commit 14c2d07
Show file tree
Hide file tree
Showing 3 changed files with 299 additions and 6 deletions.
6 changes: 3 additions & 3 deletions pkg/patterns/declarative/image.go
Original file line number Diff line number Diff line change
Expand Up @@ -29,7 +29,7 @@ import (
// ImageRegistryTransform modifies all Pods to use registry for the image source and adds the imagePullSecret
func ImageRegistryTransform(registry, imagePullSecret string) ObjectTransform {
return func(c context.Context, o DeclarativeObject, m *manifest.Objects) error {
return applyImageRegistry(c, o, m, registry, imagePullSecret, applyPrivateRegistryToImage)
return applyImageRegistry(c, m, registry, imagePullSecret, applyPrivateRegistryToImage)
}
}

Expand All @@ -38,11 +38,11 @@ type ImageFunc func(registry, image string) string
// PrivateRegistryTransform modifies all Pods to use registry for the image source and adds the imagePullSecret
func PrivateRegistryTransform(registry, imagePullSecret string, imageFunc ImageFunc) ObjectTransform {
return func(c context.Context, o DeclarativeObject, m *manifest.Objects) error {
return applyImageRegistry(c, o, m, registry, imagePullSecret, imageFunc)
return applyImageRegistry(c, m, registry, imagePullSecret, imageFunc)
}
}

func applyImageRegistry(ctx context.Context, operatorObject DeclarativeObject, manifest *manifest.Objects, registry, secret string, imageFunc ImageFunc) error {
func applyImageRegistry(ctx context.Context, manifest *manifest.Objects, registry, secret string, imageFunc ImageFunc) error {
log := log.FromContext(ctx)
if registry == "" && secret == "" {
return nil
Expand Down
280 changes: 280 additions & 0 deletions pkg/patterns/declarative/image_test.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,280 @@
/*
Copyright 2024 The Kubernetes Authors.
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/

package declarative

import (
"context"
"testing"

"github.com/google/go-cmp/cmp"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
"sigs.k8s.io/kubebuilder-declarative-pattern/pkg/patterns/declarative/pkg/manifest"
)

func Test_ImageRegistryTransform(t *testing.T) {
inputManifest := `---
apiVersion: apps/v1
kind: Deployment
metadata:
labels:
app: test-app
name: frontend
spec:
replicas: 1
selector:
matchLabels:
app: test-app
strategy: {}
template:
metadata:
labels:
app: test-app
spec:
containers:
- image: busybox
name: busybox
---
apiVersion: batch/v1
kind: CronJob
metadata:
name: hello
spec:
schedule: "* * * * *"
jobTemplate:
spec:
template:
spec:
containers:
- name: hello
image: busybox:1.28
imagePullPolicy: IfNotPresent
command:
- /bin/sh
- -c
- date; echo Hello from the Kubernetes cluster
restartPolicy: OnFailure`
var testCases = []struct {
name string
registry string
imagePullSecret string
inputManifest string
expectedManifest string
}{
{
name: "replace registry only",
registry: "gcr.io/foo/bar",
imagePullSecret: "",
inputManifest: inputManifest,
expectedManifest: `---
apiVersion: apps/v1
kind: Deployment
metadata:
labels:
app: test-app
name: frontend
spec:
replicas: 1
selector:
matchLabels:
app: test-app
strategy: {}
template:
metadata:
labels:
app: test-app
spec:
containers:
- image: gcr.io/foo/bar/busybox
name: busybox
---
apiVersion: batch/v1
kind: CronJob
metadata:
name: hello
spec:
schedule: "* * * * *"
jobTemplate:
spec:
template:
spec:
containers:
- name: hello
image: gcr.io/foo/bar/busybox:1.28
imagePullPolicy: IfNotPresent
command:
- /bin/sh
- -c
- date; echo Hello from the Kubernetes cluster
restartPolicy: OnFailure`,
},
{
name: "replace imagePullSecrets only",
registry: "",
imagePullSecret: "some-secret",
inputManifest: inputManifest,
expectedManifest: `---
apiVersion: apps/v1
kind: Deployment
metadata:
labels:
app: test-app
name: frontend
spec:
replicas: 1
selector:
matchLabels:
app: test-app
strategy: {}
template:
metadata:
labels:
app: test-app
spec:
containers:
- image: busybox
name: busybox
imagePullSecrets:
- name: some-secret
---
apiVersion: batch/v1
kind: CronJob
metadata:
name: hello
spec:
schedule: "* * * * *"
jobTemplate:
spec:
template:
spec:
containers:
- name: hello
image: busybox:1.28
imagePullPolicy: IfNotPresent
command:
- /bin/sh
- -c
- date; echo Hello from the Kubernetes cluster
imagePullSecrets:
- name: some-secret
restartPolicy: OnFailure`,
},
{
name: "replace registry and imagePullSecrets",
registry: "gcr.io/foo/bar",
imagePullSecret: "some-secret",
inputManifest: inputManifest,
expectedManifest: `---
apiVersion: apps/v1
kind: Deployment
metadata:
labels:
app: test-app
name: frontend
spec:
replicas: 1
selector:
matchLabels:
app: test-app
strategy: {}
template:
metadata:
labels:
app: test-app
spec:
containers:
- image: gcr.io/foo/bar/busybox
name: busybox
imagePullSecrets:
- name: some-secret
---
apiVersion: batch/v1
kind: CronJob
metadata:
name: hello
spec:
schedule: "* * * * *"
jobTemplate:
spec:
template:
spec:
containers:
- name: hello
image: gcr.io/foo/bar/busybox:1.28
imagePullPolicy: IfNotPresent
command:
- /bin/sh
- -c
- date; echo Hello from the Kubernetes cluster
imagePullSecrets:
- name: some-secret
restartPolicy: OnFailure`,
},
{
name: "replace nothing",
registry: "",
imagePullSecret: "",
inputManifest: inputManifest,
expectedManifest: inputManifest,
},
}
for _, tc := range testCases {
t.Run(tc.name, func(t *testing.T) {

dummyDeclarative := &TestResource{
TypeMeta: metav1.TypeMeta{
Kind: "TestResource",
APIVersion: "addons.example.org/v1alpha1",
},
ObjectMeta: metav1.ObjectMeta{
Name: "test-instance",
},
}

ctx := context.Background()

objects, err := manifest.ParseObjects(ctx, tc.inputManifest)
if err != nil {
t.Fatalf("unexpected error: %v", err)
}

fn := ImageRegistryTransform(tc.registry, tc.imagePullSecret)
err = fn(ctx, dummyDeclarative, objects)
if err != nil {
t.Fatalf("unexpected error: %v", err)
}

expectedObjects, err := manifest.ParseObjects(ctx, tc.expectedManifest)
if err != nil {
t.Fatalf("unexpected error: %v", err)
}

if len(expectedObjects.Items) != len(objects.Items) {
t.Fatal("expected number of objects does not equal number of objects")
}

for idx := range expectedObjects.Items {
diff := cmp.Diff(
expectedObjects.Items[idx].UnstructuredObject().Object,
objects.Items[idx].UnstructuredObject().Object)
if diff != "" {
t.Errorf("result mismatch (-want +got):\n%s", diff)
}
}

})
}
}
19 changes: 16 additions & 3 deletions pkg/patterns/declarative/pkg/manifest/objects.go
Original file line number Diff line number Diff line change
Expand Up @@ -161,12 +161,24 @@ func nestedFieldNoCopy(obj map[string]interface{}, fields ...string) (interface{
return val, true, nil
}

func (o *Object) podSpecPath() []string {
switch o.object.GetKind() {
case "CronJob":
return []string{"spec", "jobTemplate", "spec", "template", "spec"}
default: // Default to try the path used by common types such as Deployment, StatefulSet, etc.
return []string{"spec", "template", "spec"}
}
}

func (o *Object) MutateContainers(fn func(map[string]interface{}) error) error {
if o.object.Object == nil {
o.object.Object = make(map[string]interface{})
}

containers, found, err := nestedFieldNoCopy(o.object.Object, "spec", "template", "spec", "containers")
podSpecPath := o.podSpecPath()

containersPath := append(podSpecPath, "containers")
containers, found, err := nestedFieldNoCopy(o.object.Object, containersPath...)
if err != nil {
return fmt.Errorf("error reading containers: %v", err)
}
Expand All @@ -180,7 +192,8 @@ func (o *Object) MutateContainers(fn func(map[string]interface{}) error) error {
return fmt.Errorf("containers was not a list")
}

initContainers, found, err := nestedFieldNoCopy(o.object.Object, "spec", "template", "spec", "initContainers")
initContainersPath := append(podSpecPath, "initContainers")
initContainers, found, err := nestedFieldNoCopy(o.object.Object, initContainersPath...)
if err != nil {
return fmt.Errorf("error reading init containers: %v", err)
}
Expand Down Expand Up @@ -215,7 +228,7 @@ func (o *Object) MutatePodSpec(fn func(map[string]interface{}) error) error {
o.object.Object = make(map[string]interface{})
}

sp, found, err := nestedFieldNoCopy(o.object.Object, "spec", "template", "spec")
sp, found, err := nestedFieldNoCopy(o.object.Object, o.podSpecPath()...)
if err != nil {
return fmt.Errorf("error reading containers: %v", err)
}
Expand Down

0 comments on commit 14c2d07

Please sign in to comment.