packer-cn/command/validate.go

198 lines
4.9 KiB
Go
Raw Normal View History

package command
2013-06-13 13:03:44 -04:00
import (
"encoding/json"
2013-06-13 13:03:44 -04:00
"fmt"
"log"
"strings"
2015-05-25 20:29:10 -04:00
"github.com/hashicorp/packer/fix"
2017-04-04 16:39:01 -04:00
"github.com/hashicorp/packer/packer"
"github.com/hashicorp/packer/template"
"github.com/google/go-cmp/cmp"
"github.com/posener/complete"
2013-06-13 13:03:44 -04:00
)
type ValidateCommand struct {
Meta
2013-06-13 13:03:44 -04:00
}
func (c *ValidateCommand) Run(args []string) int {
2013-06-13 13:03:44 -04:00
var cfgSyntaxOnly bool
2015-05-25 20:29:10 -04:00
flags := c.Meta.FlagSet("validate", FlagSetBuildFilter|FlagSetVars)
flags.Usage = func() { c.Ui.Say(c.Help()) }
flags.BoolVar(&cfgSyntaxOnly, "syntax-only", false, "check syntax only")
if err := flags.Parse(args); err != nil {
return 1
}
2015-05-25 20:29:10 -04:00
args = flags.Args()
2013-06-13 13:03:44 -04:00
if len(args) != 1 {
2015-05-25 20:29:10 -04:00
flags.Usage()
2013-06-13 13:03:44 -04:00
return 1
}
2015-05-25 20:29:10 -04:00
// Parse the template
tpl, err := template.ParseFile(args[0])
if err != nil {
c.Ui.Error(fmt.Sprintf("Failed to parse template: %s", err))
return 1
}
2015-05-25 20:29:10 -04:00
// If we're only checking syntax, then we're done already
if cfgSyntaxOnly {
c.Ui.Say("Syntax-only check passed. Everything looks okay.")
return 0
}
2015-05-25 20:29:10 -04:00
// Get the core
core, err := c.Meta.Core(tpl)
2013-06-13 13:03:44 -04:00
if err != nil {
2015-05-25 20:29:10 -04:00
c.Ui.Error(err.Error())
2013-06-13 13:03:44 -04:00
return 1
}
errs := make([]error, 0)
2013-11-03 00:09:30 -04:00
warnings := make(map[string][]string)
2013-06-13 13:03:44 -04:00
2015-05-25 20:29:10 -04:00
// Get the builds we care about
buildNames := c.Meta.BuildNames(core)
builds := make([]packer.Build, 0, len(buildNames))
for _, n := range buildNames {
b, err := core.Build(n)
if err != nil {
c.Ui.Error(fmt.Sprintf(
"Failed to initialize build '%s': %s",
n, err))
return 1
2015-05-25 20:29:10 -04:00
}
2013-06-13 13:03:44 -04:00
2015-05-25 20:29:10 -04:00
builds = append(builds, b)
2013-06-13 13:03:44 -04:00
}
// Check the configuration of all builds
for _, b := range builds {
2013-06-19 00:10:34 -04:00
log.Printf("Preparing build: %s", b.Name())
2013-12-27 11:21:17 -05:00
warns, err := b.Prepare()
2013-11-03 00:09:30 -04:00
if len(warns) > 0 {
warnings[b.Name()] = warns
}
if err != nil {
errs = append(errs, fmt.Errorf("Errors validating build '%s'. %s", b.Name(), err))
}
}
2013-06-13 13:06:06 -04:00
// Check if any of the configuration is fixable
var templateData map[string]interface{}
json.Unmarshal(tpl.RawContents, &templateData)
input := make(map[string]interface{})
for k, v := range templateData {
input[k] = v
}
for _, name := range fix.FixerOrder {
var err error
fixer, ok := fix.Fixers[name]
if !ok {
panic("fixer not found: " + name)
}
input, err = fixer.Fix(input)
if err != nil {
c.Ui.Error(fmt.Sprintf("Error fixing: %s", err))
return 1
}
}
// delete empty top-level keys since the fixers seem to add them
// willy-nilly
for k := range input {
ml, ok := input[k].([]map[string]interface{})
if !ok {
continue
}
if len(ml) == 0 {
delete(input, k)
}
}
// Guaranteed to be valid json, so we can ignore errors
var fixedData map[string]interface{}
j, _ := json.Marshal(input)
json.Unmarshal(j, &fixedData)
if diff := cmp.Diff(templateData, fixedData); diff != "" {
c.Ui.Say("[warning] Fixable configuration found.")
c.Ui.Say("You may need to run `packer fix` to get your build to run")
c.Ui.Say("correctly. See debug log for more information.\n")
log.Printf("Fixable config differences:\n%s", diff)
}
2013-06-13 13:03:44 -04:00
if len(errs) > 0 {
2015-05-25 20:29:10 -04:00
c.Ui.Error("Template validation failed. Errors are shown below.\n")
for i, err := range errs {
2015-05-25 20:29:10 -04:00
c.Ui.Error(err.Error())
2013-06-13 13:24:10 -04:00
if (i + 1) < len(errs) {
2015-05-25 20:29:10 -04:00
c.Ui.Error("")
}
}
2013-06-13 13:03:44 -04:00
return 1
}
2013-11-03 00:09:30 -04:00
if len(warnings) > 0 {
2015-05-25 20:29:10 -04:00
c.Ui.Say("Template validation succeeded, but there were some warnings.")
c.Ui.Say("These are ONLY WARNINGS, and Packer will attempt to build the")
c.Ui.Say("template despite them, but they should be paid attention to.\n")
2013-11-03 00:09:30 -04:00
for build, warns := range warnings {
2015-05-25 20:29:10 -04:00
c.Ui.Say(fmt.Sprintf("Warnings for build '%s':\n", build))
2013-11-03 00:09:30 -04:00
for _, warning := range warns {
2015-05-25 20:29:10 -04:00
c.Ui.Say(fmt.Sprintf("* %s", warning))
2013-11-03 00:09:30 -04:00
}
}
return 0
}
2015-05-25 20:29:10 -04:00
c.Ui.Say("Template validated successfully.")
2013-06-13 13:03:44 -04:00
return 0
}
func (*ValidateCommand) Help() string {
helpText := `
Usage: packer validate [options] TEMPLATE
Checks the template is valid by parsing the template and also
checking the configuration with the various builders, provisioners, etc.
If it is not valid, the errors will be shown and the command will exit
with a non-zero exit status. If it is valid, it will exit with a zero
exit status.
Options:
-syntax-only Only check syntax. Do not verify config of the template.
-except=foo,bar,baz Validate all builds other than these
-only=foo,bar,baz Validate only these builds
-var 'key=value' Variable for templates, can be used multiple times.
-var-file=path JSON file containing user variables.
`
return strings.TrimSpace(helpText)
}
func (*ValidateCommand) Synopsis() string {
2013-06-13 13:03:44 -04:00
return "check that a template is valid"
}
func (*ValidateCommand) AutocompleteArgs() complete.Predictor {
return complete.PredictNothing
}
func (*ValidateCommand) AutocompleteFlags() complete.Flags {
return complete.Flags{
"-syntax-only": complete.PredictNothing,
"-except": complete.PredictNothing,
"-only": complete.PredictNothing,
"-var": complete.PredictNothing,
"-var-file": complete.PredictNothing,
}
}