Skip to content

Commit 0b3b6e9

Browse files
authored
[DENA-826] do not render Component kind of kustomization files (#42)
* do not render Component kind of kustomization files * add test for multiple kustomization fileS * run go.mod, add bunch of go cover ignores * a bit more linting, replace logs with returning error * add another go-cov ignore
1 parent cf90645 commit 0b3b6e9

File tree

5 files changed

+123
-15
lines changed

5 files changed

+123
-15
lines changed

README.md

+2
Original file line numberDiff line numberDiff line change
@@ -12,6 +12,8 @@ kustomize-build-dirs takes a list of filenames, and for each one walks up the
1212
directory tree until it finds a directory containing `kustomization.yaml` then
1313
runs `kustomize build` on that directory, saving the output in the directory
1414
given by `--out-dir`.
15+
It also truncates secrets, so that we don't need to decrypt them in order to check
16+
if manifests are correct.
1517

1618
This program should only be run from the root of a Git repository.
1719

cmd/kustomize-build-dirs/main.go

+57-2
Original file line numberDiff line numberDiff line change
@@ -3,6 +3,7 @@ package main
33
import (
44
"errors"
55
"fmt"
6+
"io"
67
"os"
78
"os/exec"
89
"path/filepath"
@@ -11,10 +12,17 @@ import (
1112

1213
"github.com/urfave/cli/v2"
1314
"golang.org/x/sync/errgroup"
15+
"gopkg.in/yaml.v2"
1416
)
1517

1618
const manifestFileName = "manifests.yaml"
1719

20+
// Kustomization represents the structure of a Kustomization file
21+
type Kustomization struct {
22+
APIVersion string `yaml:"apiVersion"`
23+
Kind string `yaml:"kind"`
24+
}
25+
1826
// variable used for testing
1927
var getwdFunc = os.Getwd
2028

@@ -66,6 +74,11 @@ func kustomizeBuildDirs(outDir string, doTruncateSecrets bool, filepaths []strin
6674
return err
6775
}
6876

77+
kustomizationRoots, err = removeComponentKustomizations(rootDir, kustomizationRoots)
78+
if err != nil { //go-cov:skip
79+
return err
80+
}
81+
6982
// truncate secrets so we can run `kustomize build` without having to decrypt them
7083
if doTruncateSecrets {
7184
if err := truncateSecrets(rootDir, kustomizationRoots); err != nil {
@@ -111,6 +124,7 @@ func findKustomizationRoots(root string, paths []string) ([]string, error) {
111124
if kustomizationRoot == "" {
112125
continue
113126
}
127+
114128
if _, exists := rootsMap[kustomizationRoot]; !exists {
115129
fmt.Printf("Found kustomization build dir: %s\n", kustomizationRoot)
116130
rootsMap[kustomizationRoot] = struct{}{}
@@ -131,8 +145,8 @@ func findKustomizationRoot(repoRoot string, relativePath string) (string, error)
131145
case err == nil:
132146
// found 'kustomization.yaml'
133147
return dir, nil
134-
case err != nil && !os.IsNotExist(err):
135-
return "", fmt.Errorf("Error checking for file in %s: %v", dir, err)
148+
case !os.IsNotExist(err):
149+
return "", fmt.Errorf("error checking for file in %s: %v", dir, err)
136150
default:
137151
// file not found, continue up the directory tree
138152
continue
@@ -141,6 +155,47 @@ func findKustomizationRoot(repoRoot string, relativePath string) (string, error)
141155
return "", nil
142156
}
143157

158+
// removeComponentKustomizations checks the list of the kustomization files, and removes those with
159+
// kind: Component.
160+
// We can't expect standalone Component kustomization files to correctly render.
161+
func removeComponentKustomizations(kustomizationRoot string, paths []string) ([]string, error) {
162+
pathsNoComponent := []string{}
163+
for _, path := range paths {
164+
isComponent, err := checkIfIsComponent(
165+
filepath.Join(kustomizationRoot, path, "kustomization.yaml"),
166+
)
167+
if err != nil { //go-cov:skip
168+
return nil, err
169+
}
170+
if !isComponent {
171+
pathsNoComponent = append(pathsNoComponent, path)
172+
}
173+
}
174+
return pathsNoComponent, nil
175+
}
176+
177+
func checkIfIsComponent(filepath string) (bool, error) {
178+
file, err := os.Open(filepath)
179+
if err != nil { //go-cov:skip
180+
return false, fmt.Errorf("failed opening kustomization file: %s: %v", filepath, err)
181+
}
182+
defer file.Close()
183+
184+
// Read the file's content
185+
data, err := io.ReadAll(file)
186+
if err != nil { //go-cov:skip
187+
return false, fmt.Errorf("error reading file: %v", err)
188+
}
189+
190+
// Unmarshal the YAML into the struct
191+
var kustomization Kustomization
192+
err = yaml.Unmarshal(data, &kustomization)
193+
if err != nil { //go-cov:skip
194+
return false, fmt.Errorf("error unmarshaling YAML: %v", err)
195+
}
196+
return kustomization.Kind == "Component", nil
197+
}
198+
144199
func truncateSecrets(rootDir string, dirs []string) error {
145200
secrets, err := findSecrets(rootDir, dirs)
146201
if err != nil {

cmd/kustomize-build-dirs/main_test.go

+54-3
Original file line numberDiff line numberDiff line change
@@ -22,6 +22,13 @@ kind: Kustomization
2222
resources:
2323
- deployment.yaml
2424
`
25+
26+
componentKustomization = `apiVersion: kustomize.config.k8s.io/v1beta1
27+
kind: Component
28+
29+
patches:
30+
- path: deployment.yaml
31+
`
2532
simpleDeploymentTemplate = `apiVersion: apps/v1
2633
kind: Deployment
2734
metadata:
@@ -44,7 +51,12 @@ func requireErorrPrefix(t *testing.T, err error, prefix string) {
4451
t.Helper()
4552

4653
require.Error(t, err)
47-
require.LessOrEqual(t, len(prefix), len(err.Error()), "error cannot be shorter than prefix")
54+
require.LessOrEqual(
55+
t,
56+
len(prefix),
57+
len(err.Error()),
58+
fmt.Sprintf("error cannot be shorter than prefix, err: %s, prefix: %s", err, prefix),
59+
)
4860
require.Equalf(t, prefix, err.Error()[:len(prefix)], "full error: %v", err)
4961
}
5062

@@ -111,7 +123,7 @@ func TestFailsWhenUnableToListSecrets(t *testing.T) {
111123
workDir,
112124
)
113125

114-
// run command outside of any Git directory
126+
// run command outside any Git directory
115127
err = kustomizeBuildDirs(mockoutDir, true, []string{"kustomization.yaml"})
116128
requireErorrPrefix(t, err, expectedErrPrefix)
117129
}
@@ -148,7 +160,7 @@ func TestFailsWhenUnableToFindKustomizations(t *testing.T) {
148160
require.NoError(t, os.Chmod(unredableDirPath, 0o600))
149161
// restore permissions so we can cleanup
150162
defer os.Chmod(unredableDirPath, 0o700) //nolint:errcheck
151-
expectedErrPrefix := "Error checking for file in manifests:"
163+
expectedErrPrefix := "error checking for file in manifests:"
152164

153165
err := kustomizeBuildDirs(mockoutDir, false, []string{"manifests/kustomization.yaml"})
154166
requireErorrPrefix(t, err, expectedErrPrefix)
@@ -305,6 +317,20 @@ func compareResults(
305317
}
306318
}
307319

320+
func TestDontRenderComponent(t *testing.T) {
321+
gitDir, outDir := setupTest(t)
322+
323+
manifestPath := filepath.Join("manifests", "deployment.yaml")
324+
repoFiles := map[string]string{
325+
filepath.Join("manifests", "kustomization.yaml"): componentKustomization,
326+
manifestPath: simpleDeployment,
327+
}
328+
buildGitRepo(t, gitDir, repoFiles)
329+
330+
require.NoError(t, kustomizeBuildDirs(outDir, false, []string{manifestPath}))
331+
require.NoFileExists(t, outDir)
332+
}
333+
308334
func TestWriteSingleManifest(t *testing.T) {
309335
gitDir, outDir := setupTest(t)
310336

@@ -367,6 +393,31 @@ func TestWriteMultipleManifests(t *testing.T) {
367393
compareResults(t, outDir, expectedContents, readOutDir(t, outDir))
368394
}
369395

396+
func TestWriteMultipleManifestsOneIscomponent(t *testing.T) {
397+
gitDir, outDir := setupTest(t)
398+
399+
firstDeploymentPath := filepath.Join("first-project", "deployment.yaml")
400+
firstDeploymentcontent := fmt.Sprintf(simpleDeploymentTemplate, "first-app")
401+
secondDeploymentPath := filepath.Join("second-project", "deployment.yaml")
402+
secondDeploymentcontent := fmt.Sprintf(simpleDeploymentTemplate, "second-app")
403+
repoFiles := map[string]string{
404+
firstDeploymentPath: firstDeploymentcontent,
405+
filepath.Join("first-project", "kustomization.yaml"): simpleKustomization,
406+
secondDeploymentPath: secondDeploymentcontent,
407+
filepath.Join("second-project", "kustomization.yaml"): componentKustomization,
408+
}
409+
buildGitRepo(t, gitDir, repoFiles)
410+
expectedContents := map[string]string{
411+
"first-project": firstDeploymentcontent,
412+
}
413+
414+
require.NoError(
415+
t,
416+
kustomizeBuildDirs(outDir, false, []string{firstDeploymentPath, secondDeploymentPath}),
417+
)
418+
compareResults(t, outDir, expectedContents, readOutDir(t, outDir))
419+
}
420+
370421
func TestSecretsStubbed(t *testing.T) {
371422
gitDir, outDir := setupTest(t)
372423
manifestsDir := filepath.Join("src", "manifests")

go.mod

+4-4
Original file line numberDiff line numberDiff line change
@@ -8,8 +8,9 @@ require (
88
github.com/urfave/cli/v2 v2.27.2
99
gitlab.com/matthewhughes/go-cov v0.4.0
1010
golang.org/x/sync v0.7.0
11-
k8s.io/apimachinery v0.30.3
12-
k8s.io/client-go v0.30.3
11+
gopkg.in/yaml.v2 v2.4.0
12+
k8s.io/apimachinery v0.30.2
13+
k8s.io/client-go v0.30.2
1314
sigs.k8s.io/controller-runtime v0.18.4
1415
)
1516

@@ -213,10 +214,9 @@ require (
213214
google.golang.org/protobuf v1.33.0 // indirect
214215
gopkg.in/inf.v0 v0.9.1 // indirect
215216
gopkg.in/ini.v1 v1.67.0 // indirect
216-
gopkg.in/yaml.v2 v2.4.0 // indirect
217217
gopkg.in/yaml.v3 v3.0.1 // indirect
218218
honnef.co/go/tools v0.4.7 // indirect
219-
k8s.io/api v0.30.3 // indirect
219+
k8s.io/api v0.30.2 // indirect
220220
k8s.io/klog/v2 v2.120.1 // indirect
221221
k8s.io/kube-openapi v0.0.0-20240228011516-70dd3763d340 // indirect
222222
k8s.io/utils v0.0.0-20230726121419-3b25d923346b // indirect

go.sum

+6-6
Original file line numberDiff line numberDiff line change
@@ -618,14 +618,14 @@ gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA=
618618
gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
619619
honnef.co/go/tools v0.4.7 h1:9MDAWxMoSnB6QoSqiVr7P5mtkT9pOc1kSxchzPCnqJs=
620620
honnef.co/go/tools v0.4.7/go.mod h1:+rnGS1THNh8zMwnd2oVOTL9QF6vmfyG6ZXBULae2uc0=
621-
k8s.io/api v0.30.3 h1:ImHwK9DCsPA9uoU3rVh4QHAHHK5dTSv1nxJUapx8hoQ=
622-
k8s.io/api v0.30.3/go.mod h1:GPc8jlzoe5JG3pb0KJCSLX5oAFIW3/qNJITlDj8BH04=
621+
k8s.io/api v0.30.2 h1:+ZhRj+28QT4UOH+BKznu4CBgPWgkXO7XAvMcMl0qKvI=
622+
k8s.io/api v0.30.2/go.mod h1:ULg5g9JvOev2dG0u2hig4Z7tQ2hHIuS+m8MNZ+X6EmI=
623623
k8s.io/apiextensions-apiserver v0.30.1 h1:4fAJZ9985BmpJG6PkoxVRpXv9vmPUOVzl614xarePws=
624624
k8s.io/apiextensions-apiserver v0.30.1/go.mod h1:R4GuSrlhgq43oRY9sF2IToFh7PVlF1JjfWdoG3pixk4=
625-
k8s.io/apimachinery v0.30.3 h1:q1laaWCmrszyQuSQCfNB8cFgCuDAoPszKY4ucAjDwHc=
626-
k8s.io/apimachinery v0.30.3/go.mod h1:iexa2somDaxdnj7bha06bhb43Zpa6eWH8N8dbqVjTUc=
627-
k8s.io/client-go v0.30.3 h1:bHrJu3xQZNXIi8/MoxYtZBBWQQXwy16zqJwloXXfD3k=
628-
k8s.io/client-go v0.30.3/go.mod h1:8d4pf8vYu665/kUbsxWAQ/JDBNWqfFeZnvFiVdmx89U=
625+
k8s.io/apimachinery v0.30.2 h1:fEMcnBj6qkzzPGSVsAZtQThU62SmQ4ZymlXRC5yFSCg=
626+
k8s.io/apimachinery v0.30.2/go.mod h1:iexa2somDaxdnj7bha06bhb43Zpa6eWH8N8dbqVjTUc=
627+
k8s.io/client-go v0.30.2 h1:sBIVJdojUNPDU/jObC+18tXWcTJVcwyqS9diGdWHk50=
628+
k8s.io/client-go v0.30.2/go.mod h1:JglKSWULm9xlJLx4KCkfLLQ7XwtlbflV6uFFSHTMgVs=
629629
k8s.io/klog/v2 v2.120.1 h1:QXU6cPEOIslTGvZaXvFWiP9VKyeet3sawzTOvdXb4Vw=
630630
k8s.io/klog/v2 v2.120.1/go.mod h1:3Jpz1GvMt720eyJH1ckRHK1EDfpxISzJ7I9OYgaDtPE=
631631
k8s.io/kube-openapi v0.0.0-20240228011516-70dd3763d340 h1:BZqlfIlq5YbRMFko6/PM7FjZpUb45WallggurYhKGag=

0 commit comments

Comments
 (0)