forked from gruntwork-io/terratest
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathhelm_basic_example_template_test.go
118 lines (98 loc) · 5.22 KB
/
helm_basic_example_template_test.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
// +build kubeall helm
// **NOTE**: we have build tags to differentiate kubernetes tests from non-kubernetes tests, and further differentiate helm
// tests. This is done because minikube is heavy and can interfere with docker related tests in terratest. Similarly, helm
// can overload the minikube system and thus interfere with the other kubernetes tests. Specifically, many of the tests
// start to fail with `connection refused` errors from `minikube`. To avoid overloading the system, we run the kubernetes
// tests and helm tests separately from the others. This may not be necessary if you have a sufficiently powerful machine.
// We recommend at least 4 cores and 16GB of RAM if you want to run all the tests together.
package test
import (
"path/filepath"
"testing"
"github.com/stretchr/testify/require"
appsv1 "k8s.io/api/apps/v1"
"github.com/gruntwork-io/terratest/modules/helm"
)
// This file contains examples of how to use terratest to test helm chart template logic by rendering the templates
// using `helm template`, and then reading in the rendered templates.
// There are two tests:
// - TestHelmBasicExampleTemplateRenderedDeployment: An example of how to read in the rendered object and check the
// computed values.
// - TestHelmBasicExampleTemplateRequiredTemplateArgs: An example of how to check that the required args are indeed
// required for the template to render.
// An example of how to verify the rendered template object of a Helm Chart given various inputs.
func TestHelmBasicExampleTemplateRenderedDeployment(t *testing.T) {
t.Parallel()
// Path to the helm chart we will test
helmChartPath, err := filepath.Abs("../examples/helm-basic-example")
require.NoError(t, err)
// Since we aren't deploying any resources, there is no need to setup kubectl authentication, helm home, or
// namespaces
// Setup the args. For this test, we will set the following input values:
// - containerImageRepo=nginx
// - containerImageTag=1.15.8
options := &helm.Options{
SetValues: map[string]string{
"containerImageRepo": "nginx",
"containerImageTag": "1.15.8",
},
}
// Run RenderTemplate to render the template and capture the output. Note that we use the version without `E`, since
// we want to assert that the template renders without any errors.
// Additionally, although we know there is only one yaml file in the template, we deliberately path a templateFiles
// arg to demonstrate how to select individual templates to render.
output := helm.RenderTemplate(t, options, helmChartPath, []string{"templates/deployment.yaml"})
// Now we use kubernetes/client-go library to render the template output into the Deployment struct. This will
// ensure the Deployment resource is rendered correctly.
var deployment appsv1.Deployment
helm.UnmarshalK8SYaml(t, output, &deployment)
// Finally, we verify the deployment pod template spec is set to the expected container image value
expectedContainerImage := "nginx:1.15.8"
deploymentContainers := deployment.Spec.Template.Spec.Containers
require.Equal(t, len(deploymentContainers), 1)
require.Equal(t, deploymentContainers[0].Image, expectedContainerImage)
}
// An example of how to verify required values for a helm chart.
func TestHelmBasicExampleTemplateRequiredTemplateArgs(t *testing.T) {
t.Parallel()
// Path to the helm chart we will test
helmChartPath, err := filepath.Abs("../examples/helm-basic-example")
require.NoError(t, err)
// Since we aren't deploying any resources, there is no need to setup kubectl authentication, helm home, or
// namespaces
// Here, we use a table driven test to iterate through all the required values as subtests. You can learn more about
// go subtests here: https://blog.golang.org/subtests
// The struct captures the inputs that we will pass to helm template and a human friendly name so we can identify it
// in the test output. In this case, each test case will be a complete values input except for one of the required
// values missing, to test that neglecting a required value will cause the template rendering to fail.
testCases := []struct {
name string
values map[string]string
}{
{
"MissingContainerImageRepo",
map[string]string{"containerImageTag": "1.15.8"},
},
{
"MissingContainerImageTag",
map[string]string{"containerImageRepo": "nginx"},
},
}
// Now we iterate over each test case and spawn a sub test
for _, testCase := range testCases {
// Here, we capture the range variable and force it into the scope of this block. If we don't do this, when the
// subtest switches contexts (because of t.Parallel), the testCase value will have been updated by the for loop
// and will be the next testCase!
testCase := testCase
// The actual sub test spawning. We name the sub test using the human friendly name. Note that we name the sub
// test T struct to subT to make it clear which T struct corresponds to which test. However, in most cases you
// will not reference the main test T so you can name it the same.
t.Run(testCase.name, func(subT *testing.T) {
subT.Parallel()
// Now we try rendering the template, but verify we get an error
options := &helm.Options{SetValues: testCase.values}
_, err := helm.RenderTemplateE(t, options, helmChartPath, []string{})
require.Error(t, err)
})
}
}