mirror of https://github.com/helm/helm
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
221 lines
7.0 KiB
221 lines
7.0 KiB
/*
|
|
Copyright The Helm 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 lint
|
|
|
|
import (
|
|
"strings"
|
|
"testing"
|
|
"time"
|
|
|
|
"helm.sh/helm/v3/pkg/chartutil"
|
|
"helm.sh/helm/v3/pkg/lint/support"
|
|
)
|
|
|
|
var values map[string]interface{}
|
|
|
|
const namespace = "testNamespace"
|
|
const strict = false
|
|
|
|
const badChartDir = "rules/testdata/badchartfile"
|
|
const badValuesFileDir = "rules/testdata/badvaluesfile"
|
|
const badYamlFileDir = "rules/testdata/albatross"
|
|
const goodChartDir = "rules/testdata/goodone"
|
|
const subChartValuesDir = "rules/testdata/withsubchart"
|
|
const malformedTemplate = "rules/testdata/malformed-template"
|
|
|
|
func TestBadChart(t *testing.T) {
|
|
m := All(badChartDir, values, namespace, strict).Messages
|
|
if len(m) != 8 {
|
|
t.Errorf("Number of errors %v", len(m))
|
|
t.Errorf("All didn't fail with expected errors, got %#v", m)
|
|
}
|
|
// There should be one INFO, and 2 ERROR messages, check for them
|
|
var i, e, e2, e3, e4, e5, e6 bool
|
|
for _, msg := range m {
|
|
if msg.Severity == support.InfoSev {
|
|
if strings.Contains(msg.Err.Error(), "icon is recommended") {
|
|
i = true
|
|
}
|
|
}
|
|
if msg.Severity == support.ErrorSev {
|
|
if strings.Contains(msg.Err.Error(), "version '0.0.0.0' is not a valid SemVer") {
|
|
e = true
|
|
}
|
|
if strings.Contains(msg.Err.Error(), "name is required") {
|
|
e2 = true
|
|
}
|
|
|
|
if strings.Contains(msg.Err.Error(), "apiVersion is required. The value must be either \"v1\" or \"v2\"") {
|
|
e3 = true
|
|
}
|
|
|
|
if strings.Contains(msg.Err.Error(), "chart type is not valid in apiVersion") {
|
|
e4 = true
|
|
}
|
|
|
|
if strings.Contains(msg.Err.Error(), "dependencies are not valid in the Chart file with apiVersion") {
|
|
e5 = true
|
|
}
|
|
// This comes from the dependency check, which loads dependency info from the Chart.yaml
|
|
if strings.Contains(msg.Err.Error(), "unable to load chart") {
|
|
e6 = true
|
|
}
|
|
}
|
|
}
|
|
if !e || !e2 || !e3 || !e4 || !e5 || !i || !e6 {
|
|
t.Errorf("Didn't find all the expected errors, got %#v", m)
|
|
}
|
|
}
|
|
|
|
func TestInvalidYaml(t *testing.T) {
|
|
m := All(badYamlFileDir, values, namespace, strict).Messages
|
|
if len(m) != 1 {
|
|
t.Fatalf("All didn't fail with expected errors, got %#v", m)
|
|
}
|
|
if !strings.Contains(m[0].Err.Error(), "deliberateSyntaxError") {
|
|
t.Errorf("All didn't have the error for deliberateSyntaxError")
|
|
}
|
|
}
|
|
|
|
func TestBadValues(t *testing.T) {
|
|
m := All(badValuesFileDir, values, namespace, strict).Messages
|
|
if len(m) < 1 {
|
|
t.Fatalf("All didn't fail with expected errors, got %#v", m)
|
|
}
|
|
if !strings.Contains(m[0].Err.Error(), "unable to parse YAML") {
|
|
t.Errorf("All didn't have the error for invalid key format: %s", m[0].Err)
|
|
}
|
|
}
|
|
|
|
func TestGoodChart(t *testing.T) {
|
|
m := All(goodChartDir, values, namespace, strict).Messages
|
|
if len(m) != 0 {
|
|
t.Error("All returned linter messages when it shouldn't have")
|
|
for i, msg := range m {
|
|
t.Logf("Message %d: %s", i, msg)
|
|
}
|
|
}
|
|
}
|
|
|
|
// TestHelmCreateChart tests that a `helm create` always passes a `helm lint` test.
|
|
//
|
|
// See https://github.com/helm/helm/issues/7923
|
|
func TestHelmCreateChart(t *testing.T) {
|
|
dir := t.TempDir()
|
|
|
|
createdChart, err := chartutil.Create("testhelmcreatepasseslint", dir)
|
|
if err != nil {
|
|
t.Error(err)
|
|
// Fatal is bad because of the defer.
|
|
return
|
|
}
|
|
|
|
// Note: we test with strict=true here, even though others have
|
|
// strict = false.
|
|
m := All(createdChart, values, namespace, true).Messages
|
|
if ll := len(m); ll != 1 {
|
|
t.Errorf("All should have had exactly 1 error. Got %d", ll)
|
|
for i, msg := range m {
|
|
t.Logf("Message %d: %s", i, msg.Error())
|
|
}
|
|
} else if msg := m[0].Err.Error(); !strings.Contains(msg, "icon is recommended") {
|
|
t.Errorf("Unexpected lint error: %s", msg)
|
|
}
|
|
}
|
|
|
|
// TestHelmCreateChart_CheckDeprecatedWarnings checks if any default template created by `helm create` throws
|
|
// deprecated warnings in the linter check against the current Kubernetes version (provided using ldflags).
|
|
//
|
|
// See https://github.com/helm/helm/issues/11495
|
|
//
|
|
// Resources like hpa and ingress, which are disabled by default in values.yaml are enabled here using the equivalent
|
|
// of the `--set` flag.
|
|
//
|
|
// Note: This test requires the following ldflags to be set per the current Kubernetes version to avoid false-positive
|
|
// results.
|
|
// 1. -X helm.sh/helm/v3/pkg/lint/rules.k8sVersionMajor=<k8s-major-version>
|
|
// 2. -X helm.sh/helm/v3/pkg/lint/rules.k8sVersionMinor=<k8s-minor-version>
|
|
// or directly use '$(LDFLAGS)' in Makefile.
|
|
//
|
|
// When run without ldflags, the test passes giving a false-positive result. This is because the variables
|
|
// `k8sVersionMajor` and `k8sVersionMinor` by default are set to an older version of Kubernetes, with which, there
|
|
// might not be the deprecation warning.
|
|
func TestHelmCreateChart_CheckDeprecatedWarnings(t *testing.T) {
|
|
createdChart, err := chartutil.Create("checkdeprecatedwarnings", t.TempDir())
|
|
if err != nil {
|
|
t.Error(err)
|
|
return
|
|
}
|
|
|
|
// Add values to enable hpa, and ingress which are disabled by default.
|
|
// This is the equivalent of:
|
|
// helm lint checkdeprecatedwarnings --set 'autoscaling.enabled=true,ingress.enabled=true'
|
|
updatedValues := map[string]interface{}{
|
|
"autoscaling": map[string]interface{}{
|
|
"enabled": true,
|
|
},
|
|
"ingress": map[string]interface{}{
|
|
"enabled": true,
|
|
},
|
|
}
|
|
|
|
linterRunDetails := All(createdChart, updatedValues, namespace, true)
|
|
for _, msg := range linterRunDetails.Messages {
|
|
if strings.HasPrefix(msg.Error(), "[WARNING]") &&
|
|
strings.Contains(msg.Error(), "deprecated") {
|
|
// When there is a deprecation warning for an object created
|
|
// by `helm create` for the current Kubernetes version, fail.
|
|
t.Errorf("Unexpected deprecation warning for %q: %s", msg.Path, msg.Error())
|
|
}
|
|
}
|
|
}
|
|
|
|
// lint ignores import-values
|
|
// See https://github.com/helm/helm/issues/9658
|
|
func TestSubChartValuesChart(t *testing.T) {
|
|
m := All(subChartValuesDir, values, namespace, strict).Messages
|
|
if len(m) != 0 {
|
|
t.Error("All returned linter messages when it shouldn't have")
|
|
for i, msg := range m {
|
|
t.Logf("Message %d: %s", i, msg)
|
|
}
|
|
}
|
|
}
|
|
|
|
// lint stuck with malformed template object
|
|
// See https://github.com/helm/helm/issues/11391
|
|
func TestMalformedTemplate(t *testing.T) {
|
|
c := time.After(3 * time.Second)
|
|
ch := make(chan int, 1)
|
|
var m []support.Message
|
|
go func() {
|
|
m = All(malformedTemplate, values, namespace, strict).Messages
|
|
ch <- 1
|
|
}()
|
|
select {
|
|
case <-c:
|
|
t.Fatalf("lint malformed template timeout")
|
|
case <-ch:
|
|
if len(m) != 1 {
|
|
t.Fatalf("All didn't fail with expected errors, got %#v", m)
|
|
}
|
|
if !strings.Contains(m[0].Err.Error(), "invalid character '{'") {
|
|
t.Errorf("All didn't have the error for invalid character '{'")
|
|
}
|
|
}
|
|
}
|