/ *
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 rules
import (
"os"
"path/filepath"
"regexp"
"github.com/pkg/errors"
"sigs.k8s.io/yaml"
"helm.sh/helm/v3/pkg/chart/loader"
"helm.sh/helm/v3/pkg/chartutil"
"helm.sh/helm/v3/pkg/engine"
"helm.sh/helm/v3/pkg/lint/support"
)
var (
crdHookSearch = regexp . MustCompile ( ` "?helm\.sh/hook"?:\s+crd-install ` )
releaseTimeSearch = regexp . MustCompile ( ` \.Release\.Time ` )
)
// Templates lints the templates in the Linter.
func Templates ( linter * support . Linter , values map [ string ] interface { } , namespace string , strict bool ) {
path := "templates/"
templatesPath := filepath . Join ( linter . ChartDir , path )
templatesDirExist := linter . RunLinterRule ( support . WarningSev , path , validateTemplatesDir ( templatesPath ) )
// Templates directory is optional for now
if ! templatesDirExist {
return
}
// Load chart and parse templates, based on tiller/release_server
chart , err := loader . Load ( linter . ChartDir )
chartLoaded := linter . RunLinterRule ( support . ErrorSev , path , err )
if ! chartLoaded {
return
}
options := chartutil . ReleaseOptions {
Name : "testRelease" ,
Namespace : namespace ,
}
cvals , err := chartutil . CoalesceValues ( chart , values )
if err != nil {
return
}
valuesToRender , err := chartutil . ToRenderValues ( chart , cvals , options , nil )
if err != nil {
linter . RunLinterRule ( support . ErrorSev , path , err )
return
}
var e engine . Engine
e . Strict = strict
e . LintMode = true
renderedContentMap , err := e . Render ( chart , valuesToRender )
renderOk := linter . RunLinterRule ( support . ErrorSev , path , err )
if ! renderOk {
return
}
/ * Iterate over all the templates to check :
- It is a . yaml file
- All the values in the template file is defined
- { { } } include | quote
- Generated content is a valid Yaml file
- Metadata . Namespace is not set
* /
for _ , template := range chart . Templates {
fileName , data := template . Name , template . Data
path = fileName
linter . RunLinterRule ( support . ErrorSev , path , validateAllowedExtension ( fileName ) )
// These are v3 specific checks to make sure and warn people if their
// chart is not compatible with v3
linter . RunLinterRule ( support . WarningSev , path , validateNoCRDHooks ( data ) )
linter . RunLinterRule ( support . ErrorSev , path , validateNoReleaseTime ( data ) )
// We only apply the following lint rules to yaml files
if filepath . Ext ( fileName ) != ".yaml" || filepath . Ext ( fileName ) == ".yml" {
continue
}
// NOTE: disabled for now, Refs https://github.com/helm/helm/issues/1463
// Check that all the templates have a matching value
//linter.RunLinterRule(support.WarningSev, path, validateNoMissingValues(templatesPath, valuesToRender, preExecutedTemplate))
// NOTE: disabled for now, Refs https://github.com/helm/helm/issues/1037
// linter.RunLinterRule(support.WarningSev, path, validateQuotes(string(preExecutedTemplate)))
renderedContent := renderedContentMap [ filepath . Join ( chart . Name ( ) , fileName ) ]
var yamlStruct K8sYamlStruct
// Even though K8sYamlStruct only defines Metadata namespace, an error in any other
// key will be raised as well
err := yaml . Unmarshal ( [ ] byte ( renderedContent ) , & yamlStruct )
// If YAML linting fails, we sill progress. So we don't capture the returned state
// on this linter run.
linter . RunLinterRule ( support . ErrorSev , path , validateYamlContent ( err ) )
}
}
// Validation functions
func validateTemplatesDir ( templatesPath string ) error {
if fi , err := os . Stat ( templatesPath ) ; err != nil {
return errors . New ( "directory not found" )
} else if ! fi . IsDir ( ) {
return errors . New ( "not a directory" )
}
return nil
}
func validateAllowedExtension ( fileName string ) error {
ext := filepath . Ext ( fileName )
validExtensions := [ ] string { ".yaml" , ".yml" , ".tpl" , ".txt" }
for _ , b := range validExtensions {
if b == ext {
return nil
}
}
return errors . Errorf ( "file extension '%s' not valid. Valid extensions are .yaml, .yml, .tpl, or .txt" , ext )
}
func validateYamlContent ( err error ) error {
return errors . Wrap ( err , "unable to parse YAML" )
}
func validateNoCRDHooks ( manifest [ ] byte ) error {
if crdHookSearch . Match ( manifest ) {
return errors . New ( "manifest is a crd-install hook. This hook is no longer supported in v3 and all CRDs should also exist the crds/ directory at the top level of the chart" )
}
return nil
}
func validateNoReleaseTime ( manifest [ ] byte ) error {
if releaseTimeSearch . Match ( manifest ) {
return errors . New ( ".Release.Time has been removed in v3, please replace with the `now` function in your templates" )
}
return nil
}
// K8sYamlStruct stubs a Kubernetes YAML file.
// Need to access for now to Namespace only
type K8sYamlStruct struct {
Metadata struct {
Namespace string
}
}