/* 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= // 2. -X helm.sh/helm/v3/pkg/lint/rules.k8sVersionMinor= // 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 '{'") } } }