Merge pull request #8882 from hashicorp/fix-var-file-hcl

allow to use hcl files as var files in HCL mode
This commit is contained in:
Megan Marsh 2020-03-13 12:35:58 -07:00 committed by GitHub
commit 6477d8a0c8
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
13 changed files with 150 additions and 34 deletions

View File

@ -104,7 +104,7 @@ func (c *BuildCommand) GetBuildsFromHCL(path string) ([]packer.Build, int) {
PostProcessorsSchemas: c.CoreConfig.Components.PostProcessorStore, PostProcessorsSchemas: c.CoreConfig.Components.PostProcessorStore,
} }
builds, diags := parser.Parse(path, c.flagVars) builds, diags := parser.Parse(path, c.varFiles, c.flagVars)
{ {
// write HCL errors/diagnostics if any. // write HCL errors/diagnostics if any.
b := bytes.NewBuffer(nil) b := bytes.NewBuffer(nil)

View File

@ -32,6 +32,7 @@ type Meta struct {
Version string Version string
// These are set by command-line flags // These are set by command-line flags
varFiles []string
flagVars map[string]string flagVars map[string]string
} }
@ -41,6 +42,17 @@ func (m *Meta) Core(tpl *template.Template) (*packer.Core, error) {
// Copy the config so we don't modify it // Copy the config so we don't modify it
config := *m.CoreConfig config := *m.CoreConfig
config.Template = tpl config.Template = tpl
fj := &kvflag.FlagJSON{}
for _, file := range m.varFiles {
err := fj.Set(file)
if err != nil {
return nil, err
}
}
for k, v := range *fj {
m.flagVars[k] = v
}
config.Variables = m.flagVars config.Variables = m.flagVars
// Init the core // Init the core
@ -117,7 +129,7 @@ func (m *Meta) FlagSet(n string, fs FlagSetFlags) *flag.FlagSet {
// FlagSetVars tells us what variables to use // FlagSetVars tells us what variables to use
if fs&FlagSetVars != 0 { if fs&FlagSetVars != 0 {
f.Var((*kvflag.Flag)(&m.flagVars), "var", "") f.Var((*kvflag.Flag)(&m.flagVars), "var", "")
f.Var((*kvflag.FlagJSON)(&m.flagVars), "var-file", "") f.Var((*kvflag.StringSlice)(&m.varFiles), "var-file", "")
} }
// Create an io.Writer that writes to our Ui properly for errors. // Create an io.Writer that writes to our Ui properly for errors.

View File

@ -37,6 +37,7 @@ func getBasicParser() *Parser {
type parseTestArgs struct { type parseTestArgs struct {
filename string filename string
vars map[string]string vars map[string]string
varFiles []string
} }
type parseTest struct { type parseTest struct {
@ -58,7 +59,7 @@ func testParse(t *testing.T, tests []parseTest) {
for _, tt := range tests { for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) { t.Run(tt.name, func(t *testing.T) {
gotCfg, gotDiags := tt.parser.parse(tt.args.filename, tt.args.vars) gotCfg, gotDiags := tt.parser.parse(tt.args.filename, tt.args.varFiles, tt.args.vars)
if tt.parseWantDiags == (gotDiags == nil) { if tt.parseWantDiags == (gotDiags == nil) {
t.Fatalf("Parser.parse() unexpected %q diagnostics.", gotDiags) t.Fatalf("Parser.parse() unexpected %q diagnostics.", gotDiags)
} }
@ -87,8 +88,11 @@ func testParse(t *testing.T, tests []parseTest) {
gotInputVar := gotCfg.InputVariables gotInputVar := gotCfg.InputVariables
for name, value := range tt.parseWantCfg.InputVariables { for name, value := range tt.parseWantCfg.InputVariables {
if variable, ok := gotInputVar[name]; ok { if variable, ok := gotInputVar[name]; ok {
if variable.DefaultValue.GoString() != value.DefaultValue.GoString() { if diff := cmp.Diff(variable.DefaultValue.GoString(), value.DefaultValue.GoString()); diff != "" {
t.Fatalf("Parser.parse() input variable %s expected '%s' but was '%s'", name, value.DefaultValue.GoString(), variable.DefaultValue.GoString()) t.Fatalf("Parser.parse(): unexpected default value for %s: %s", name, diff)
}
if diff := cmp.Diff(variable.VarfileValue.GoString(), value.VarfileValue.GoString()); diff != "" {
t.Fatalf("Parser.parse(): varfile value differs for %s: %s", name, diff)
} }
} else { } else {
t.Fatalf("Parser.parse() missing input variable. %s", name) t.Fatalf("Parser.parse() missing input variable. %s", name)

View File

@ -52,7 +52,7 @@ const (
hcl2VarJsonFileExt = ".auto.pkrvars.json" hcl2VarJsonFileExt = ".auto.pkrvars.json"
) )
func (p *Parser) parse(filename string, vars map[string]string) (*PackerConfig, hcl.Diagnostics) { func (p *Parser) parse(filename string, varFiles []string, argVars map[string]string) (*PackerConfig, hcl.Diagnostics) {
var files []*hcl.File var files []*hcl.File
var diags hcl.Diagnostics var diags hcl.Diagnostics
@ -60,6 +60,7 @@ func (p *Parser) parse(filename string, vars map[string]string) (*PackerConfig,
// parse config files // parse config files
{ {
hclFiles, jsonFiles, moreDiags := GetHCL2Files(filename, hcl2FileExt, hcl2JsonFileExt) hclFiles, jsonFiles, moreDiags := GetHCL2Files(filename, hcl2FileExt, hcl2JsonFileExt)
diags = append(diags, moreDiags...)
if len(hclFiles)+len(jsonFiles) == 0 { if len(hclFiles)+len(jsonFiles) == 0 {
diags = append(moreDiags, &hcl.Diagnostic{ diags = append(moreDiags, &hcl.Diagnostic{
Severity: hcl.DiagError, Severity: hcl.DiagError,
@ -111,6 +112,20 @@ func (p *Parser) parse(filename string, vars map[string]string) (*PackerConfig,
{ {
hclVarFiles, jsonVarFiles, moreDiags := GetHCL2Files(filename, hcl2VarFileExt, hcl2VarJsonFileExt) hclVarFiles, jsonVarFiles, moreDiags := GetHCL2Files(filename, hcl2VarFileExt, hcl2VarJsonFileExt)
diags = append(diags, moreDiags...) diags = append(diags, moreDiags...)
for _, file := range varFiles {
switch filepath.Ext(file) {
case ".hcl":
hclVarFiles = append(hclVarFiles, file)
case ".json":
jsonVarFiles = append(jsonVarFiles, file)
default:
diags = append(moreDiags, &hcl.Diagnostic{
Severity: hcl.DiagError,
Summary: "Could not guess format of " + file,
Detail: "A var file must be suffixed with `.hcl` or `.json`.",
})
}
}
var varFiles []*hcl.File var varFiles []*hcl.File
for _, filename := range hclVarFiles { for _, filename := range hclVarFiles {
f, moreDiags := p.ParseHCLFile(filename) f, moreDiags := p.ParseHCLFile(filename)
@ -123,7 +138,7 @@ func (p *Parser) parse(filename string, vars map[string]string) (*PackerConfig,
varFiles = append(varFiles, f) varFiles = append(varFiles, f)
} }
diags = append(diags, cfg.collectInputVariableValues(os.Environ(), varFiles, vars)...) diags = append(diags, cfg.collectInputVariableValues(os.Environ(), varFiles, argVars)...)
} }
_, moreDiags := cfg.InputVariables.Values() _, moreDiags := cfg.InputVariables.Values()

View File

@ -0,0 +1,4 @@
variable "foo" {
type = string
default = "bar"
}

View File

@ -0,0 +1 @@
foo = "wee"

View File

@ -13,7 +13,7 @@ func TestParse_build(t *testing.T) {
tests := []parseTest{ tests := []parseTest{
{"basic build no src", {"basic build no src",
defaultParser, defaultParser,
parseTestArgs{"testdata/build/basic.pkr.hcl", nil}, parseTestArgs{"testdata/build/basic.pkr.hcl", nil, nil},
&PackerConfig{ &PackerConfig{
Basedir: filepath.Join("testdata", "build"), Basedir: filepath.Join("testdata", "build"),
Builds: Builds{ Builds: Builds{
@ -47,7 +47,7 @@ func TestParse_build(t *testing.T) {
}, },
{"untyped provisioner", {"untyped provisioner",
defaultParser, defaultParser,
parseTestArgs{"testdata/build/provisioner_untyped.pkr.hcl", nil}, parseTestArgs{"testdata/build/provisioner_untyped.pkr.hcl", nil, nil},
&PackerConfig{ &PackerConfig{
Basedir: filepath.Join("testdata", "build"), Basedir: filepath.Join("testdata", "build"),
Builds: nil, Builds: nil,
@ -58,7 +58,7 @@ func TestParse_build(t *testing.T) {
}, },
{"inexistent provisioner", {"inexistent provisioner",
defaultParser, defaultParser,
parseTestArgs{"testdata/build/provisioner_inexistent.pkr.hcl", nil}, parseTestArgs{"testdata/build/provisioner_inexistent.pkr.hcl", nil, nil},
&PackerConfig{ &PackerConfig{
Basedir: filepath.Join("testdata", "build"), Basedir: filepath.Join("testdata", "build"),
Builds: nil, Builds: nil,
@ -69,7 +69,7 @@ func TestParse_build(t *testing.T) {
}, },
{"untyped post-processor", {"untyped post-processor",
defaultParser, defaultParser,
parseTestArgs{"testdata/build/post-processor_untyped.pkr.hcl", nil}, parseTestArgs{"testdata/build/post-processor_untyped.pkr.hcl", nil, nil},
&PackerConfig{ &PackerConfig{
Basedir: filepath.Join("testdata", "build"), Basedir: filepath.Join("testdata", "build"),
Builds: nil, Builds: nil,
@ -80,7 +80,7 @@ func TestParse_build(t *testing.T) {
}, },
{"inexistent post-processor", {"inexistent post-processor",
defaultParser, defaultParser,
parseTestArgs{"testdata/build/post-processor_inexistent.pkr.hcl", nil}, parseTestArgs{"testdata/build/post-processor_inexistent.pkr.hcl", nil, nil},
&PackerConfig{ &PackerConfig{
Basedir: filepath.Join("testdata", "build"), Basedir: filepath.Join("testdata", "build"),
Builds: nil, Builds: nil,
@ -91,7 +91,7 @@ func TestParse_build(t *testing.T) {
}, },
{"invalid source", {"invalid source",
defaultParser, defaultParser,
parseTestArgs{"testdata/build/invalid_source_reference.pkr.hcl", nil}, parseTestArgs{"testdata/build/invalid_source_reference.pkr.hcl", nil, nil},
&PackerConfig{ &PackerConfig{
Basedir: filepath.Join("testdata", "build"), Basedir: filepath.Join("testdata", "build"),
Builds: nil, Builds: nil,

View File

@ -287,8 +287,8 @@ func (p *Parser) getBuilds(cfg *PackerConfig) ([]packer.Build, hcl.Diagnostics)
// //
// Parse then return a slice of packer.Builds; which are what packer core uses // Parse then return a slice of packer.Builds; which are what packer core uses
// to run builds. // to run builds.
func (p *Parser) Parse(path string, vars map[string]string) ([]packer.Build, hcl.Diagnostics) { func (p *Parser) Parse(path string, varFiles []string, argVars map[string]string) ([]packer.Build, hcl.Diagnostics) {
cfg, diags := p.parse(path, vars) cfg, diags := p.parse(path, varFiles, argVars)
if diags.HasErrors() { if diags.HasErrors() {
return nil, diags return nil, diags
} }

View File

@ -18,7 +18,7 @@ func TestParser_complete(t *testing.T) {
tests := []parseTest{ tests := []parseTest{
{"working build", {"working build",
defaultParser, defaultParser,
parseTestArgs{"testdata/complete", nil}, parseTestArgs{"testdata/complete", nil, nil},
&PackerConfig{ &PackerConfig{
Basedir: "testdata/complete", Basedir: "testdata/complete",
InputVariables: Variables{ InputVariables: Variables{
@ -128,7 +128,7 @@ func TestParser_complete(t *testing.T) {
}, },
{"dir with no config files", {"dir with no config files",
defaultParser, defaultParser,
parseTestArgs{"testdata/empty", nil}, parseTestArgs{"testdata/empty", nil, nil},
nil, nil,
true, true, true, true,
nil, nil,
@ -136,14 +136,14 @@ func TestParser_complete(t *testing.T) {
}, },
{name: "inexistent dir", {name: "inexistent dir",
parser: defaultParser, parser: defaultParser,
args: parseTestArgs{"testdata/inexistent", nil}, args: parseTestArgs{"testdata/inexistent", nil, nil},
parseWantCfg: nil, parseWantCfg: nil,
parseWantDiags: true, parseWantDiags: true,
parseWantDiagHasErrors: true, parseWantDiagHasErrors: true,
}, },
{name: "folder named build.pkr.hcl with an unknown src", {name: "folder named build.pkr.hcl with an unknown src",
parser: defaultParser, parser: defaultParser,
args: parseTestArgs{"testdata/build.pkr.hcl", nil}, args: parseTestArgs{"testdata/build.pkr.hcl", nil, nil},
parseWantCfg: &PackerConfig{ parseWantCfg: &PackerConfig{
Basedir: "testdata/build.pkr.hcl", Basedir: "testdata/build.pkr.hcl",
Builds: Builds{ Builds: Builds{
@ -166,7 +166,7 @@ func TestParser_complete(t *testing.T) {
}, },
{name: "unknown block type", {name: "unknown block type",
parser: defaultParser, parser: defaultParser,
args: parseTestArgs{"testdata/unknown", nil}, args: parseTestArgs{"testdata/unknown", nil, nil},
parseWantCfg: &PackerConfig{ parseWantCfg: &PackerConfig{
Basedir: "testdata/unknown", Basedir: "testdata/unknown",
}, },

View File

@ -13,7 +13,7 @@ func TestParse_source(t *testing.T) {
tests := []parseTest{ tests := []parseTest{
{"two basic sources", {"two basic sources",
defaultParser, defaultParser,
parseTestArgs{"testdata/sources/basic.pkr.hcl", nil}, parseTestArgs{"testdata/sources/basic.pkr.hcl", nil, nil},
&PackerConfig{ &PackerConfig{
Basedir: filepath.Join("testdata", "sources"), Basedir: filepath.Join("testdata", "sources"),
Sources: map[SourceRef]*SourceBlock{ Sources: map[SourceRef]*SourceBlock{
@ -32,7 +32,7 @@ func TestParse_source(t *testing.T) {
}, },
{"untyped source", {"untyped source",
defaultParser, defaultParser,
parseTestArgs{"testdata/sources/untyped.pkr.hcl", nil}, parseTestArgs{"testdata/sources/untyped.pkr.hcl", nil, nil},
&PackerConfig{ &PackerConfig{
Basedir: filepath.Join("testdata", "sources"), Basedir: filepath.Join("testdata", "sources"),
}, },
@ -42,7 +42,7 @@ func TestParse_source(t *testing.T) {
}, },
{"unnamed source", {"unnamed source",
defaultParser, defaultParser,
parseTestArgs{"testdata/sources/unnamed.pkr.hcl", nil}, parseTestArgs{"testdata/sources/unnamed.pkr.hcl", nil, nil},
&PackerConfig{ &PackerConfig{
Basedir: filepath.Join("testdata", "sources"), Basedir: filepath.Join("testdata", "sources"),
}, },
@ -52,7 +52,7 @@ func TestParse_source(t *testing.T) {
}, },
{"inexistent source", {"inexistent source",
defaultParser, defaultParser,
parseTestArgs{"testdata/sources/inexistent.pkr.hcl", nil}, parseTestArgs{"testdata/sources/inexistent.pkr.hcl", nil, nil},
&PackerConfig{ &PackerConfig{
Basedir: filepath.Join("testdata", "sources"), Basedir: filepath.Join("testdata", "sources"),
}, },
@ -62,7 +62,7 @@ func TestParse_source(t *testing.T) {
}, },
{"duplicate source", {"duplicate source",
defaultParser, defaultParser,
parseTestArgs{"testdata/sources/duplicate.pkr.hcl", nil}, parseTestArgs{"testdata/sources/duplicate.pkr.hcl", nil, nil},
&PackerConfig{ &PackerConfig{
Basedir: filepath.Join("testdata", "sources"), Basedir: filepath.Join("testdata", "sources"),
Sources: map[SourceRef]*SourceBlock{ Sources: map[SourceRef]*SourceBlock{

View File

@ -20,7 +20,7 @@ func TestParse_variables(t *testing.T) {
tests := []parseTest{ tests := []parseTest{
{"basic variables", {"basic variables",
defaultParser, defaultParser,
parseTestArgs{"testdata/variables/basic.pkr.hcl", nil}, parseTestArgs{"testdata/variables/basic.pkr.hcl", nil, nil},
&PackerConfig{ &PackerConfig{
Basedir: filepath.Join("testdata", "variables"), Basedir: filepath.Join("testdata", "variables"),
InputVariables: Variables{ InputVariables: Variables{
@ -75,7 +75,7 @@ func TestParse_variables(t *testing.T) {
}, },
{"duplicate variable", {"duplicate variable",
defaultParser, defaultParser,
parseTestArgs{"testdata/variables/duplicate_variable.pkr.hcl", nil}, parseTestArgs{"testdata/variables/duplicate_variable.pkr.hcl", nil, nil},
&PackerConfig{ &PackerConfig{
Basedir: filepath.Join("testdata", "variables"), Basedir: filepath.Join("testdata", "variables"),
InputVariables: Variables{ InputVariables: Variables{
@ -90,7 +90,7 @@ func TestParse_variables(t *testing.T) {
}, },
{"duplicate variable in variables", {"duplicate variable in variables",
defaultParser, defaultParser,
parseTestArgs{"testdata/variables/duplicate_variables.pkr.hcl", nil}, parseTestArgs{"testdata/variables/duplicate_variables.pkr.hcl", nil, nil},
&PackerConfig{ &PackerConfig{
Basedir: filepath.Join("testdata", "variables"), Basedir: filepath.Join("testdata", "variables"),
InputVariables: Variables{ InputVariables: Variables{
@ -105,7 +105,7 @@ func TestParse_variables(t *testing.T) {
}, },
{"invalid default type", {"invalid default type",
defaultParser, defaultParser,
parseTestArgs{"testdata/variables/invalid_default.pkr.hcl", nil}, parseTestArgs{"testdata/variables/invalid_default.pkr.hcl", nil, nil},
&PackerConfig{ &PackerConfig{
Basedir: filepath.Join("testdata", "variables"), Basedir: filepath.Join("testdata", "variables"),
InputVariables: Variables{ InputVariables: Variables{
@ -121,7 +121,7 @@ func TestParse_variables(t *testing.T) {
{"unknown key", {"unknown key",
defaultParser, defaultParser,
parseTestArgs{"testdata/variables/unknown_key.pkr.hcl", nil}, parseTestArgs{"testdata/variables/unknown_key.pkr.hcl", nil, nil},
&PackerConfig{ &PackerConfig{
Basedir: filepath.Join("testdata", "variables"), Basedir: filepath.Join("testdata", "variables"),
InputVariables: Variables{ InputVariables: Variables{
@ -138,7 +138,7 @@ func TestParse_variables(t *testing.T) {
{"unset used variable", {"unset used variable",
defaultParser, defaultParser,
parseTestArgs{"testdata/variables/unset_used_string_variable.pkr.hcl", nil}, parseTestArgs{"testdata/variables/unset_used_string_variable.pkr.hcl", nil, nil},
&PackerConfig{ &PackerConfig{
Basedir: filepath.Join("testdata", "variables"), Basedir: filepath.Join("testdata", "variables"),
InputVariables: Variables{ InputVariables: Variables{
@ -154,7 +154,7 @@ func TestParse_variables(t *testing.T) {
{"unset unused variable", {"unset unused variable",
defaultParser, defaultParser,
parseTestArgs{"testdata/variables/unset_unused_string_variable.pkr.hcl", nil}, parseTestArgs{"testdata/variables/unset_unused_string_variable.pkr.hcl", nil, nil},
&PackerConfig{ &PackerConfig{
Basedir: filepath.Join("testdata", "variables"), Basedir: filepath.Join("testdata", "variables"),
InputVariables: Variables{ InputVariables: Variables{
@ -189,7 +189,7 @@ func TestParse_variables(t *testing.T) {
{"locals within another locals usage in different files", {"locals within another locals usage in different files",
defaultParser, defaultParser,
parseTestArgs{"testdata/variables/complicated", nil}, parseTestArgs{"testdata/variables/complicated", nil, nil},
&PackerConfig{ &PackerConfig{
Basedir: "testdata/variables/complicated", Basedir: "testdata/variables/complicated",
InputVariables: Variables{ InputVariables: Variables{
@ -231,7 +231,7 @@ func TestParse_variables(t *testing.T) {
}, },
{"recursive locals", {"recursive locals",
defaultParser, defaultParser,
parseTestArgs{"testdata/variables/recursive_locals.pkr.hcl", nil}, parseTestArgs{"testdata/variables/recursive_locals.pkr.hcl", nil, nil},
&PackerConfig{ &PackerConfig{
Basedir: filepath.Join("testdata", "variables"), Basedir: filepath.Join("testdata", "variables"),
LocalVariables: Variables{}, LocalVariables: Variables{},
@ -240,6 +240,35 @@ func TestParse_variables(t *testing.T) {
[]packer.Build{}, []packer.Build{},
false, false,
}, },
{"set variable from var-file",
defaultParser,
parseTestArgs{"testdata/variables/foo-string.variable.pkr.hcl", nil, []string{"testdata/variables/set-foo-too-wee.hcl"}},
&PackerConfig{
Basedir: filepath.Join("testdata", "variables"),
InputVariables: Variables{
"foo": &Variable{
DefaultValue: cty.StringVal("bar"),
Name: "foo",
VarfileValue: cty.StringVal("wee"),
},
},
},
false, false,
[]packer.Build{},
false,
},
{"unknown variable from var-file",
defaultParser,
parseTestArgs{"testdata/variables/empty.pkr.hcl", nil, []string{"testdata/variables/set-foo-too-wee.hcl"}},
&PackerConfig{
Basedir: filepath.Join("testdata", "variables"),
},
true, false,
[]packer.Build{},
false,
},
} }
testParse(t, tests) testParse(t, tests)
} }

View File

@ -0,0 +1,16 @@
package kvflag
import (
"strings"
)
type StringSlice []string
func (s *StringSlice) String() string {
return strings.Join(*s, ", ")
}
func (s *StringSlice) Set(value string) error {
*s = append(*s, value)
return nil
}

View File

@ -0,0 +1,35 @@
package kvflag
import (
"testing"
"github.com/google/go-cmp/cmp"
)
func TestStringSlice_Set(t *testing.T) {
type args struct {
values []string
}
tests := []struct {
name string
s StringSlice
args args
wantStringSlice StringSlice
}{
{"basic", StringSlice{"hey", "yo"}, args{[]string{"how", "are", "you"}},
StringSlice{"hey", "yo", "how", "are", "you"}},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
for _, value := range tt.args.values {
err := tt.s.Set(value)
if err != nil {
t.Fatal(err)
}
}
if diff := cmp.Diff(tt.s, tt.wantStringSlice); diff != "" {
t.Fatal(diff)
}
})
}
}