blob: 97be7d5e8690c568a0e5a3a8f4515c1c128f562e [file] [log] [blame]
Mohammed Naser3dfc2a82024-02-20 23:51:39 -05001package charts
2
3import (
4 "io"
5 "os"
6 "strings"
7 "testing"
8
9 "github.com/stretchr/testify/assert"
10 "github.com/stretchr/testify/require"
11 "github.com/yannh/kubeconform/pkg/validator"
12 "helm.sh/helm/v3/pkg/action"
13 "helm.sh/helm/v3/pkg/chart/loader"
14 "helm.sh/helm/v3/pkg/chartutil"
15)
16
17var (
18 KUBERNETES_VERSIONS = []string{
Mohammed Naser3dfc2a82024-02-20 23:51:39 -050019 "1.24.0",
20 "1.25.0",
21 "1.26.0",
22 "1.27.0",
23 "1.28.0",
24 }
25)
26
27func TestKubeconform(t *testing.T) {
28 t.Parallel()
29
30 files, err := os.ReadDir("./")
31 require.NoError(t, err)
32 require.NotEmpty(t, files)
33
34 schemas := []string{
35 "https://raw.githubusercontent.com/yannh/kubernetes-json-schema/master/{{ .NormalizedKubernetesVersion }}-standalone{{ .StrictSuffix }}/{{ .ResourceKind }}{{ .KindSuffix }}.json",
36 "https://raw.githubusercontent.com/datreeio/CRDs-catalog/main/{{.Group}}/{{.ResourceKind}}_{{.ResourceAPIVersion}}.json",
37 }
38
39 var clients map[string]*action.Install = make(map[string]*action.Install)
40 for _, version := range KUBERNETES_VERSIONS {
41 kubeVersion, err := chartutil.ParseKubeVersion(version)
42 require.NoError(t, err)
43
44 client := action.NewInstall(&action.Configuration{})
45 client.ClientOnly = true
46 client.DryRun = true
47 client.ReleaseName = "kubeconform"
48 client.Namespace = "default"
49 client.IncludeCRDs = true
50 client.KubeVersion = kubeVersion
51
52 clients[version] = client
53 }
54
55 var validators map[string]validator.Validator = make(map[string]validator.Validator)
56 for _, version := range KUBERNETES_VERSIONS {
57 opts := validator.Opts{
58 KubernetesVersion: version,
59 SkipKinds: map[string]struct{}{
Oleksandr K.99651a62024-10-30 04:41:51 +010060 "CephBlockPool": {},
61 "CephCluster": {},
62 "CephFilesystem": {},
63 "CephObjectStore": {},
64 "CephFilesystemSubVolumeGroup": {},
Mohammed Naser3dfc2a82024-02-20 23:51:39 -050065 "apiextensions.k8s.io/v1/CustomResourceDefinition": {},
66 },
67 Strict: true,
68 }
69
70 v, err := validator.New(schemas, opts)
71 require.NoError(t, err)
72
73 validators[version] = v
74 }
75
76 for _, file := range files {
77 if !file.IsDir() {
78 continue
79 }
Oleksandr K.99651a62024-10-30 04:41:51 +010080 if file.Name() == "patches" {
81 continue
82 }
Mohammed Naser3dfc2a82024-02-20 23:51:39 -050083
84 t.Run(file.Name(), func(t *testing.T) {
85 chart, err := loader.LoadDir(file.Name())
86 require.NoError(t, err)
87
88 t.Parallel()
89
90 for _, version := range KUBERNETES_VERSIONS {
91 t.Run(version, func(t *testing.T) {
92 client := clients[version]
93 v := validators[version]
94
95 t.Parallel()
96
Oleksandr K.99651a62024-10-30 04:41:51 +010097 rel, err := client.Run(
98 chart,
99 // NOTE(okozachenko1203): loki helm chart default values doesn't work.
100 map[string]interface{}{
101 "loki": map[string]interface{}{
102 "storage": map[string]interface{}{
103 "bucketNames": map[string]string{
104 "chunks": "FIXME",
105 "ruler": "FIXME",
106 "admin": "FIXME",
107 },
108 },
109 "useTestSchema": true,
110 },
111 },
112 )
Mohammed Naser3dfc2a82024-02-20 23:51:39 -0500113 require.NoError(t, err)
114
115 manifests := io.NopCloser(strings.NewReader(rel.Manifest))
116 for _, res := range v.Validate(chart.Name(), manifests) {
117 require.NoError(t, res.Err)
118 assert.Empty(t, res.ValidationErrors)
119 }
120 })
121 }
122 })
123 }
124}