Skip to content

Commit

Permalink
feat: add configmap validation
Browse files Browse the repository at this point in the history
  • Loading branch information
smlx committed Nov 1, 2021
1 parent 480f641 commit 3453a47
Show file tree
Hide file tree
Showing 5 changed files with 76 additions and 6 deletions.
5 changes: 3 additions & 2 deletions cmd/lagoon-linter/main.go
Original file line number Diff line number Diff line change
Expand Up @@ -11,8 +11,9 @@ var (

// CLI represents the command-line interface.
type CLI struct {
Validate ValidateCmd `kong:"cmd,default=1,help='(default) Validate the Lagoon YAML'"`
Version VersionCmd `kong:"cmd,help='Print version information'"`
Validate ValidateCmd `kong:"cmd,default=1,help='(default) Validate the Lagoon YAML'"`
Version VersionCmd `kong:"cmd,help='Print version information'"`
ValidateConfigMapJSON ValidateConfigMapJSONCmd `kong:"cmd,help='Validate the result of: kubectl get configmap -A -o json'"`
}

func main() {
Expand Down
2 changes: 1 addition & 1 deletion cmd/lagoon-linter/validate.go
Original file line number Diff line number Diff line change
Expand Up @@ -9,5 +9,5 @@ type ValidateCmd struct {

// Run the validation of the Lagoon YAML.
func (cmd *ValidateCmd) Run() error {
return lagoonyml.Lint(cmd.LagoonYAML, lagoonyml.RouteAnnotation())
return lagoonyml.LintFile(cmd.LagoonYAML, lagoonyml.RouteAnnotation())
}
50 changes: 50 additions & 0 deletions cmd/lagoon-linter/validateconfigmapjson.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,50 @@
package main

import (
"encoding/json"
"fmt"
"os"

"github.com/uselagoon/lagoon-linter/internal/lagoonyml"
)

// ValidateConfigMapJSONCmd represents the validate command.
type ValidateConfigMapJSONCmd struct {
ConfigMapJSON string `kong:"default='configmap.json',help='Specify the configmap JSON file dump.'"`
}

// ConfigMap represents an individual configmap.
type ConfigMap struct {
Data map[string]string `json:"data"`
Metadata map[string]interface{} `json:"metadata"`
}

// ConfigMapList represents a list of configmaps.
type ConfigMapList struct {
ConfigMaps []ConfigMap `json:"items"`
}

// Run the validation of the Lagoon YAML dumps.
func (cmd *ValidateConfigMapJSONCmd) Run() error {
// open the file
rawJSON, err := os.ReadFile(cmd.ConfigMapJSON)
if err != nil {
return fmt.Errorf("couldn't read file: %v", err)
}
var cml ConfigMapList
// unmarshal ConfigMapList
if err := json.Unmarshal(rawJSON, &cml); err != nil {
return fmt.Errorf("couldn't unmarshal JSON: %v", err)
}
// lint it
for _, cm := range cml.ConfigMaps {
if lagoonYAML, ok := cm.Data[".lagoon.yml"]; ok {
err := lagoonyml.LintYAML([]byte(lagoonYAML),
lagoonyml.RouteAnnotation())
if err != nil {
fmt.Printf("bad .lagoon.yml: %s: %v\n", cm.Metadata["namespace"], err)
}
}
}
return nil
}
23 changes: 21 additions & 2 deletions internal/lagoonyml/lint.go
Original file line number Diff line number Diff line change
Expand Up @@ -10,11 +10,11 @@ import (
// Linter validates the given Lagoon struct.
type Linter func(*Lagoon) error

// Lint takes a file path, reads it, and applies `.lagoon.yml` lint policy to
// LintFile takes a file path, reads it, and applies `.lagoon.yml` lint policy to
// it. Lint returns an error of type ErrLint if it finds problems with the
// file, a regular error if something else went wrong, and nil if the
// `.lagoon.yml` is valid.
func Lint(path string, linters ...Linter) error {
func LintFile(path string, linters ...Linter) error {
var l Lagoon
rawYAML, err := os.ReadFile(path)
if err != nil {
Expand All @@ -33,3 +33,22 @@ func Lint(path string, linters ...Linter) error {
}
return nil
}

// LintYAML takes a byte slice containing raw YAML and applies `.lagoon.yml` lint policy to
// it. Lint returns an error of type ErrLint if it finds problems with the
// file, a regular error if something else went wrong, and nil if the
// `.lagoon.yml` is valid.
func LintYAML(rawYAML []byte, linters ...Linter) error {
var l Lagoon
if err := yaml.Unmarshal(rawYAML, &l); err != nil {
return fmt.Errorf("couldn't unmarshal YAML: %v", err)
}
for _, linter := range linters {
if err := linter(&l); err != nil {
return &ErrLint{
Detail: err.Error(),
}
}
}
return nil
}
2 changes: 1 addition & 1 deletion internal/lagoonyml/lint_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -46,7 +46,7 @@ func TestLint(t *testing.T) {
}
for name, tc := range testCases {
t.Run(name, func(tt *testing.T) {
err := lagoonyml.Lint(tc.input, lagoonyml.RouteAnnotation())
err := lagoonyml.LintFile(tc.input, lagoonyml.RouteAnnotation())
if tc.valid {
if err != nil {
tt.Fatalf("unexpected error %v", err)
Expand Down

0 comments on commit 3453a47

Please sign in to comment.