Merge pull request #8482 from hashicorp/b-output-dup

config: variable names and outputs must be unique
This commit is contained in:
Mitchell Hashimoto 2016-08-26 13:57:59 -07:00 committed by GitHub
commit 706b2e2aea
9 changed files with 109 additions and 35 deletions

View File

@ -239,6 +239,12 @@ func (c *Config) Validate() error {
vars := c.InterpolatedVariables() vars := c.InterpolatedVariables()
varMap := make(map[string]*Variable) varMap := make(map[string]*Variable)
for _, v := range c.Variables { for _, v := range c.Variables {
if _, ok := varMap[v.Name]; ok {
errs = append(errs, fmt.Errorf(
"Variable '%s': duplicate found. Variable names must be unique.",
v.Name))
}
varMap[v.Name] = v varMap[v.Name] = v
} }
@ -586,7 +592,18 @@ func (c *Config) Validate() error {
} }
// Check that all outputs are valid // Check that all outputs are valid
{
found := make(map[string]struct{})
for _, o := range c.Outputs { for _, o := range c.Outputs {
// Verify the output is new
if _, ok := found[o.Name]; ok {
errs = append(errs, fmt.Errorf(
"%s: duplicate output. output names must be unique.",
o.Name))
continue
}
found[o.Name] = struct{}{}
var invalidKeys []string var invalidKeys []string
valueKeyFound := false valueKeyFound := false
for k := range o.RawConfig.Raw { for k := range o.RawConfig.Raw {
@ -626,6 +643,7 @@ func (c *Config) Validate() error {
} }
} }
} }
}
// Check that all variables are in the proper context // Check that all variables are in the proper context
for source, rc := range c.rawConfigs() { for source, rc := range c.rawConfigs() {

View File

@ -300,6 +300,13 @@ func TestConfigValidate_outputBadField(t *testing.T) {
} }
} }
func TestConfigValidate_outputDuplicate(t *testing.T) {
c := testConfig(t, "validate-output-dup")
if err := c.Validate(); err == nil {
t.Fatal("should not be valid")
}
}
func TestConfigValidate_pathVar(t *testing.T) { func TestConfigValidate_pathVar(t *testing.T) {
c := testConfig(t, "validate-path-var") c := testConfig(t, "validate-path-var")
if err := c.Validate(); err != nil { if err := c.Validate(); err != nil {
@ -440,6 +447,13 @@ func TestConfigValidate_varDefaultInterpolate(t *testing.T) {
} }
} }
func TestConfigValidate_varDup(t *testing.T) {
c := testConfig(t, "validate-var-dup")
if err := c.Validate(); err == nil {
t.Fatal("should not be valid")
}
}
func TestConfigValidate_varMultiExactNonSlice(t *testing.T) { func TestConfigValidate_varMultiExactNonSlice(t *testing.T) {
c := testConfig(t, "validate-var-multi-exact-non-slice") c := testConfig(t, "validate-var-multi-exact-non-slice")
if err := c.Validate(); err != nil { if err := c.Validate(); err != nil {

View File

@ -29,6 +29,7 @@ func (t *hclConfigurable) Config() (*Config, error) {
} }
type hclVariable struct { type hclVariable struct {
Name string `hcl:",key"`
Default interface{} Default interface{}
Description string Description string
DeclaredType string `hcl:"type"` DeclaredType string `hcl:"type"`
@ -36,7 +37,7 @@ func (t *hclConfigurable) Config() (*Config, error) {
} }
var rawConfig struct { var rawConfig struct {
Variable map[string]*hclVariable Variable []*hclVariable
} }
// Top-level item should be the object list // Top-level item should be the object list
@ -56,7 +57,7 @@ func (t *hclConfigurable) Config() (*Config, error) {
config := new(Config) config := new(Config)
if len(rawConfig.Variable) > 0 { if len(rawConfig.Variable) > 0 {
config.Variables = make([]*Variable, 0, len(rawConfig.Variable)) config.Variables = make([]*Variable, 0, len(rawConfig.Variable))
for k, v := range rawConfig.Variable { for _, v := range rawConfig.Variable {
// Defaults turn into a slice of map[string]interface{} and // Defaults turn into a slice of map[string]interface{} and
// we need to make sure to convert that down into the // we need to make sure to convert that down into the
// proper type for Config. // proper type for Config.
@ -72,7 +73,7 @@ func (t *hclConfigurable) Config() (*Config, error) {
} }
newVar := &Variable{ newVar := &Variable{
Name: k, Name: v.Name,
DeclaredType: v.DeclaredType, DeclaredType: v.DeclaredType,
Default: v.Default, Default: v.Default,
Description: v.Description, Description: v.Description,

View File

@ -462,6 +462,22 @@ func TestLoadDir_override(t *testing.T) {
} }
} }
func TestLoadDir_overrideVar(t *testing.T) {
c, err := LoadDir(filepath.Join(fixtureDir, "dir-override-var"))
if err != nil {
t.Fatalf("err: %s", err)
}
if c == nil {
t.Fatal("config should not be nil")
}
actual := variablesStr(c.Variables)
if actual != strings.TrimSpace(dirOverrideVarsVariablesStr) {
t.Fatalf("bad:\n%s", actual)
}
}
func TestLoadFile_mismatchedVariableTypes(t *testing.T) { func TestLoadFile_mismatchedVariableTypes(t *testing.T) {
_, err := LoadFile(filepath.Join(fixtureDir, "variable-mismatched-type.tf")) _, err := LoadFile(filepath.Join(fixtureDir, "variable-mismatched-type.tf"))
if err == nil { if err == nil {
@ -943,6 +959,12 @@ foo
bar bar
` `
const dirOverrideVarsVariablesStr = `
foo
baz
bar
`
const importProvidersStr = ` const importProvidersStr = `
aws aws
bar bar

View File

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

View File

@ -0,0 +1,3 @@
variable "foo" {
default = "baz"
}

View File

@ -0,0 +1,10 @@
resource "aws_instance" "web" {
}
output "ip" {
value = "foo"
}
output "ip" {
value = "bar"
}

View File

@ -2,7 +2,7 @@ variable "foo" {
default = "bar" default = "bar"
} }
variable "foo" { variable "foo2" {
default = { default = {
foo = "bar" foo = "bar"
} }

View File

@ -0,0 +1,2 @@
variable "foo" {}
variable "foo" {}