packer: required user variables [GH-374]

This commit is contained in:
Mitchell Hashimoto 2013-08-31 17:33:17 -07:00
parent 13198eb570
commit fd4b01cf85
5 changed files with 119 additions and 19 deletions

View File

@ -1,5 +1,10 @@
## 0.3.6 (unreleased) ## 0.3.6 (unreleased)
FEATURES:
* User variables can now be specified as "required", meaning the user
MUST specify a value. Just set the default value to "null". [GH-374]
IMPROVEMENTS: IMPROVEMENTS:
* core: Much improved interrupt handling. For example, interrupts now * core: Much improved interrupt handling. For example, interrupts now

View File

@ -77,7 +77,7 @@ type coreBuild struct {
hooks map[string][]Hook hooks map[string][]Hook
postProcessors [][]coreBuildPostProcessor postProcessors [][]coreBuildPostProcessor
provisioners []coreBuildProvisioner provisioners []coreBuildProvisioner
variables map[string]string variables map[string]coreBuildVariable
debug bool debug bool
force bool force bool
@ -101,6 +101,12 @@ type coreBuildProvisioner struct {
config []interface{} config []interface{}
} }
// A user-variable that is part of a single build.
type coreBuildVariable struct {
Default string
Required bool
}
// Returns the name of the build. // Returns the name of the build.
func (b *coreBuild) Name() string { func (b *coreBuild) Name() string {
return b.name return b.name
@ -122,25 +128,41 @@ func (b *coreBuild) Prepare(userVars map[string]string) (err error) {
// Compile the variables // Compile the variables
variables := make(map[string]string) variables := make(map[string]string)
for k, v := range b.variables { for k, v := range b.variables {
variables[k] = v if !v.Required {
variables[k] = v.Default
}
} }
varErrs := make([]error, 0)
if userVars != nil { if userVars != nil {
errs := make([]error, 0)
for k, v := range userVars { for k, v := range userVars {
if _, ok := variables[k]; !ok { if _, ok := variables[k]; !ok {
errs = append( varErrs = append(
errs, fmt.Errorf("Unknown user variable: %s", k)) varErrs, fmt.Errorf("Unknown user variable: %s", k))
continue continue
} }
variables[k] = v variables[k] = v
} }
}
if len(errs) > 0 { // Verify all required variables have been set.
return &MultiError{ for k, v := range b.variables {
Errors: errs, if !v.Required {
} continue
}
if _, ok := variables[k]; !ok {
varErrs = append(
varErrs, fmt.Errorf("Required user variable '%s' not set", k))
}
}
// If there were any problem with variables, return an error right
// away because we can't be certain anything else will actually work.
if len(varErrs) > 0 {
return &MultiError{
Errors: varErrs,
} }
} }

View File

@ -23,7 +23,7 @@ func testBuild() *coreBuild {
coreBuildPostProcessor{&TestPostProcessor{artifactId: "pp"}, "testPP", make(map[string]interface{}), true}, coreBuildPostProcessor{&TestPostProcessor{artifactId: "pp"}, "testPP", make(map[string]interface{}), true},
}, },
}, },
variables: make(map[string]string), variables: make(map[string]coreBuildVariable),
} }
} }
@ -116,7 +116,7 @@ func TestBuildPrepare_variables_default(t *testing.T) {
} }
build := testBuild() build := testBuild()
build.variables["foo"] = "bar" build.variables["foo"] = coreBuildVariable{Default: "bar"}
builder := build.builder.(*TestBuilder) builder := build.builder.(*TestBuilder)
err := build.Prepare(nil) err := build.Prepare(nil)
@ -135,7 +135,7 @@ func TestBuildPrepare_variables_default(t *testing.T) {
func TestBuildPrepare_variables_nonexist(t *testing.T) { func TestBuildPrepare_variables_nonexist(t *testing.T) {
build := testBuild() build := testBuild()
build.variables["foo"] = "bar" build.variables["foo"] = coreBuildVariable{Default: "bar"}
err := build.Prepare(map[string]string{"bar": "baz"}) err := build.Prepare(map[string]string{"bar": "baz"})
if err == nil { if err == nil {
@ -150,7 +150,7 @@ func TestBuildPrepare_variables_override(t *testing.T) {
} }
build := testBuild() build := testBuild()
build.variables["foo"] = "bar" build.variables["foo"] = coreBuildVariable{Default: "bar"}
builder := build.builder.(*TestBuilder) builder := build.builder.(*TestBuilder)
err := build.Prepare(map[string]string{"foo": "baz"}) err := build.Prepare(map[string]string{"foo": "baz"})
@ -167,6 +167,16 @@ func TestBuildPrepare_variables_override(t *testing.T) {
} }
} }
func TestBuildPrepare_variablesRequired(t *testing.T) {
build := testBuild()
build.variables["foo"] = coreBuildVariable{Required: true}
err := build.Prepare(map[string]string{})
if err == nil {
t.Fatal("should have had error")
}
}
func TestBuild_Run(t *testing.T) { func TestBuild_Run(t *testing.T) {
assert := asserts.NewTestingAsserts(t, true) assert := asserts.NewTestingAsserts(t, true)

View File

@ -16,7 +16,7 @@ import (
// "interface{}" pointers since we actually don't know what their contents // "interface{}" pointers since we actually don't know what their contents
// are until we read the "type" field. // are until we read the "type" field.
type rawTemplate struct { type rawTemplate struct {
Variables map[string]string Variables map[string]interface{}
Builders []map[string]interface{} Builders []map[string]interface{}
Hooks map[string][]string Hooks map[string][]string
Provisioners []map[string]interface{} Provisioners []map[string]interface{}
@ -26,7 +26,7 @@ type rawTemplate struct {
// The Template struct represents a parsed template, parsed into the most // The Template struct represents a parsed template, parsed into the most
// completed form it can be without additional processing by the caller. // completed form it can be without additional processing by the caller.
type Template struct { type Template struct {
Variables map[string]string Variables map[string]RawVariable
Builders map[string]RawBuilderConfig Builders map[string]RawBuilderConfig
Hooks map[string][]string Hooks map[string][]string
PostProcessors [][]RawPostProcessorConfig PostProcessors [][]RawPostProcessorConfig
@ -63,6 +63,12 @@ type RawProvisionerConfig struct {
RawConfig interface{} RawConfig interface{}
} }
// RawVariable represents a variable configuration within a template.
type RawVariable struct {
Default string
Required bool
}
// ParseTemplate takes a byte slice and parses a Template from it, returning // ParseTemplate takes a byte slice and parses a Template from it, returning
// the template and possibly errors while loading the template. The error // the template and possibly errors while loading the template. The error
// could potentially be a MultiError, representing multiple errors. Knowing // could potentially be a MultiError, representing multiple errors. Knowing
@ -105,7 +111,7 @@ func ParseTemplate(data []byte) (t *Template, err error) {
} }
t = &Template{} t = &Template{}
t.Variables = make(map[string]string) t.Variables = make(map[string]RawVariable)
t.Builders = make(map[string]RawBuilderConfig) t.Builders = make(map[string]RawBuilderConfig)
t.Hooks = rawTpl.Hooks t.Hooks = rawTpl.Hooks
t.PostProcessors = make([][]RawPostProcessorConfig, len(rawTpl.PostProcessors)) t.PostProcessors = make([][]RawPostProcessorConfig, len(rawTpl.PostProcessors))
@ -113,7 +119,22 @@ func ParseTemplate(data []byte) (t *Template, err error) {
// Gather all the variables // Gather all the variables
for k, v := range rawTpl.Variables { for k, v := range rawTpl.Variables {
t.Variables[k] = v var variable RawVariable
variable.Default = ""
variable.Required = v == nil
if v != nil {
def, ok := v.(string)
if !ok {
errors = append(errors,
fmt.Errorf("variable '%s': default value must be string or null", k))
continue
}
variable.Default = def
}
t.Variables[k] = variable
} }
// Gather all the builders // Gather all the builders
@ -428,6 +449,15 @@ func (t *Template) Build(name string, components *ComponentFinder) (b Build, err
provisioners = append(provisioners, coreProv) provisioners = append(provisioners, coreProv)
} }
// Prepare the variables
variables := make(map[string]coreBuildVariable)
for k, v := range t.Variables {
variables[k] = coreBuildVariable{
Default: v.Default,
Required: v.Required,
}
}
b = &coreBuild{ b = &coreBuild{
name: name, name: name,
builder: builder, builder: builder,
@ -436,7 +466,7 @@ func (t *Template) Build(name string, components *ComponentFinder) (b Build, err
hooks: hooks, hooks: hooks,
postProcessors: postProcessors, postProcessors: postProcessors,
provisioners: provisioners, provisioners: provisioners,
variables: t.Variables, variables: variables,
} }
return return

View File

@ -364,7 +364,7 @@ func TestParseTemplate_Variables(t *testing.T) {
{ {
"variables": { "variables": {
"foo": "bar", "foo": "bar",
"bar": "" "bar": null
}, },
"builders": [{"type": "something"}] "builders": [{"type": "something"}]
@ -379,6 +379,39 @@ func TestParseTemplate_Variables(t *testing.T) {
if result.Variables == nil || len(result.Variables) != 2 { if result.Variables == nil || len(result.Variables) != 2 {
t.Fatalf("bad vars: %#v", result.Variables) t.Fatalf("bad vars: %#v", result.Variables)
} }
if result.Variables["foo"].Default != "bar" {
t.Fatal("foo default is not right")
}
if result.Variables["foo"].Required {
t.Fatal("foo should not be required")
}
if result.Variables["bar"].Default != "" {
t.Fatal("default should be empty")
}
if !result.Variables["bar"].Required {
t.Fatal("bar should be required")
}
}
func TestParseTemplate_variablesBadDefault(t *testing.T) {
data := `
{
"variables": {
"foo": 7,
},
"builders": [{"type": "something"}]
}
`
_, err := ParseTemplate([]byte(data))
if err == nil {
t.Fatal("should have error")
}
} }
func TestTemplate_BuildNames(t *testing.T) { func TestTemplate_BuildNames(t *testing.T) {