2014-07-21 02:17:03 +02:00
|
|
|
package config
|
|
|
|
|
|
|
|
import (
|
|
|
|
"reflect"
|
|
|
|
"testing"
|
|
|
|
)
|
|
|
|
|
|
|
|
func TestMerge(t *testing.T) {
|
|
|
|
cases := []struct {
|
|
|
|
c1, c2, result *Config
|
|
|
|
err bool
|
|
|
|
}{
|
|
|
|
// Normal good case.
|
|
|
|
{
|
|
|
|
&Config{
|
2015-03-05 21:56:31 +01:00
|
|
|
Atlas: &AtlasConfig{
|
|
|
|
Name: "foo",
|
|
|
|
},
|
2014-09-12 04:54:02 +02:00
|
|
|
Modules: []*Module{
|
|
|
|
&Module{Name: "foo"},
|
|
|
|
},
|
2014-07-21 02:17:03 +02:00
|
|
|
Outputs: []*Output{
|
|
|
|
&Output{Name: "foo"},
|
|
|
|
},
|
|
|
|
ProviderConfigs: []*ProviderConfig{
|
|
|
|
&ProviderConfig{Name: "foo"},
|
|
|
|
},
|
|
|
|
Resources: []*Resource{
|
|
|
|
&Resource{Name: "foo"},
|
|
|
|
},
|
|
|
|
Variables: []*Variable{
|
|
|
|
&Variable{Name: "foo"},
|
|
|
|
},
|
|
|
|
|
|
|
|
unknownKeys: []string{"foo"},
|
|
|
|
},
|
|
|
|
|
|
|
|
&Config{
|
2015-03-05 21:56:31 +01:00
|
|
|
Atlas: &AtlasConfig{
|
|
|
|
Name: "bar",
|
|
|
|
},
|
2014-09-12 04:54:02 +02:00
|
|
|
Modules: []*Module{
|
|
|
|
&Module{Name: "bar"},
|
|
|
|
},
|
2014-07-21 02:17:03 +02:00
|
|
|
Outputs: []*Output{
|
|
|
|
&Output{Name: "bar"},
|
|
|
|
},
|
|
|
|
ProviderConfigs: []*ProviderConfig{
|
|
|
|
&ProviderConfig{Name: "bar"},
|
|
|
|
},
|
|
|
|
Resources: []*Resource{
|
|
|
|
&Resource{Name: "bar"},
|
|
|
|
},
|
|
|
|
Variables: []*Variable{
|
|
|
|
&Variable{Name: "bar"},
|
|
|
|
},
|
|
|
|
|
|
|
|
unknownKeys: []string{"bar"},
|
|
|
|
},
|
|
|
|
|
|
|
|
&Config{
|
2015-03-05 21:56:31 +01:00
|
|
|
Atlas: &AtlasConfig{
|
|
|
|
Name: "bar",
|
|
|
|
},
|
2014-09-12 04:54:02 +02:00
|
|
|
Modules: []*Module{
|
|
|
|
&Module{Name: "foo"},
|
|
|
|
&Module{Name: "bar"},
|
|
|
|
},
|
2014-07-21 02:17:03 +02:00
|
|
|
Outputs: []*Output{
|
|
|
|
&Output{Name: "foo"},
|
|
|
|
&Output{Name: "bar"},
|
|
|
|
},
|
|
|
|
ProviderConfigs: []*ProviderConfig{
|
|
|
|
&ProviderConfig{Name: "foo"},
|
|
|
|
&ProviderConfig{Name: "bar"},
|
|
|
|
},
|
|
|
|
Resources: []*Resource{
|
|
|
|
&Resource{Name: "foo"},
|
|
|
|
&Resource{Name: "bar"},
|
|
|
|
},
|
|
|
|
Variables: []*Variable{
|
|
|
|
&Variable{Name: "foo"},
|
|
|
|
&Variable{Name: "bar"},
|
|
|
|
},
|
|
|
|
|
|
|
|
unknownKeys: []string{"foo", "bar"},
|
|
|
|
},
|
|
|
|
|
|
|
|
false,
|
|
|
|
},
|
|
|
|
|
|
|
|
// Test that when merging duplicates, it merges into the
|
|
|
|
// first, but keeps the duplicates so that errors still
|
|
|
|
// happen.
|
|
|
|
{
|
|
|
|
&Config{
|
|
|
|
Outputs: []*Output{
|
|
|
|
&Output{Name: "foo"},
|
|
|
|
},
|
|
|
|
ProviderConfigs: []*ProviderConfig{
|
|
|
|
&ProviderConfig{Name: "foo"},
|
|
|
|
},
|
|
|
|
Resources: []*Resource{
|
|
|
|
&Resource{Name: "foo"},
|
|
|
|
},
|
|
|
|
Variables: []*Variable{
|
|
|
|
&Variable{Name: "foo", Default: "foo"},
|
|
|
|
&Variable{Name: "foo"},
|
|
|
|
},
|
|
|
|
|
|
|
|
unknownKeys: []string{"foo"},
|
|
|
|
},
|
|
|
|
|
|
|
|
&Config{
|
|
|
|
Outputs: []*Output{
|
|
|
|
&Output{Name: "bar"},
|
|
|
|
},
|
|
|
|
ProviderConfigs: []*ProviderConfig{
|
|
|
|
&ProviderConfig{Name: "bar"},
|
|
|
|
},
|
|
|
|
Resources: []*Resource{
|
|
|
|
&Resource{Name: "bar"},
|
|
|
|
},
|
|
|
|
Variables: []*Variable{
|
2014-07-21 16:32:36 +02:00
|
|
|
&Variable{Name: "foo", Default: "bar"},
|
2014-07-21 02:17:03 +02:00
|
|
|
&Variable{Name: "bar"},
|
|
|
|
},
|
|
|
|
|
|
|
|
unknownKeys: []string{"bar"},
|
|
|
|
},
|
|
|
|
|
|
|
|
&Config{
|
|
|
|
Outputs: []*Output{
|
|
|
|
&Output{Name: "foo"},
|
|
|
|
&Output{Name: "bar"},
|
|
|
|
},
|
|
|
|
ProviderConfigs: []*ProviderConfig{
|
|
|
|
&ProviderConfig{Name: "foo"},
|
|
|
|
&ProviderConfig{Name: "bar"},
|
|
|
|
},
|
|
|
|
Resources: []*Resource{
|
|
|
|
&Resource{Name: "foo"},
|
|
|
|
&Resource{Name: "bar"},
|
|
|
|
},
|
|
|
|
Variables: []*Variable{
|
2014-07-21 16:32:36 +02:00
|
|
|
&Variable{Name: "foo", Default: "bar"},
|
2014-07-21 02:17:03 +02:00
|
|
|
&Variable{Name: "foo"},
|
|
|
|
&Variable{Name: "bar"},
|
|
|
|
},
|
|
|
|
|
|
|
|
unknownKeys: []string{"foo", "bar"},
|
|
|
|
},
|
|
|
|
|
|
|
|
false,
|
|
|
|
},
|
2016-12-14 06:48:59 +01:00
|
|
|
|
|
|
|
// Terraform block
|
|
|
|
{
|
|
|
|
&Config{
|
|
|
|
Terraform: &Terraform{
|
|
|
|
RequiredVersion: "A",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
&Config{},
|
|
|
|
&Config{
|
|
|
|
Terraform: &Terraform{
|
|
|
|
RequiredVersion: "A",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
false,
|
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
&Config{},
|
|
|
|
&Config{
|
|
|
|
Terraform: &Terraform{
|
|
|
|
RequiredVersion: "A",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
&Config{
|
|
|
|
Terraform: &Terraform{
|
|
|
|
RequiredVersion: "A",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
false,
|
|
|
|
},
|
|
|
|
|
|
|
|
// Provider alias
|
|
|
|
{
|
|
|
|
&Config{
|
|
|
|
ProviderConfigs: []*ProviderConfig{
|
|
|
|
&ProviderConfig{Alias: "foo"},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
&Config{},
|
|
|
|
&Config{
|
|
|
|
ProviderConfigs: []*ProviderConfig{
|
|
|
|
&ProviderConfig{Alias: "foo"},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
false,
|
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
&Config{},
|
|
|
|
&Config{
|
|
|
|
ProviderConfigs: []*ProviderConfig{
|
|
|
|
&ProviderConfig{Alias: "foo"},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
&Config{
|
|
|
|
ProviderConfigs: []*ProviderConfig{
|
|
|
|
&ProviderConfig{Alias: "foo"},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
false,
|
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
&Config{
|
|
|
|
ProviderConfigs: []*ProviderConfig{
|
|
|
|
&ProviderConfig{Alias: "bar"},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
&Config{
|
|
|
|
ProviderConfigs: []*ProviderConfig{
|
|
|
|
&ProviderConfig{Alias: "foo"},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
&Config{
|
|
|
|
ProviderConfigs: []*ProviderConfig{
|
|
|
|
&ProviderConfig{Alias: "foo"},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
false,
|
|
|
|
},
|
|
|
|
|
|
|
|
// Variable type
|
|
|
|
{
|
|
|
|
&Config{
|
|
|
|
Variables: []*Variable{
|
|
|
|
&Variable{DeclaredType: "foo"},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
&Config{},
|
|
|
|
&Config{
|
|
|
|
Variables: []*Variable{
|
|
|
|
&Variable{DeclaredType: "foo"},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
false,
|
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
&Config{},
|
|
|
|
&Config{
|
|
|
|
Variables: []*Variable{
|
|
|
|
&Variable{DeclaredType: "foo"},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
&Config{
|
|
|
|
Variables: []*Variable{
|
|
|
|
&Variable{DeclaredType: "foo"},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
false,
|
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
&Config{
|
|
|
|
Variables: []*Variable{
|
|
|
|
&Variable{DeclaredType: "bar"},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
&Config{
|
|
|
|
Variables: []*Variable{
|
|
|
|
&Variable{DeclaredType: "foo"},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
&Config{
|
|
|
|
Variables: []*Variable{
|
|
|
|
&Variable{DeclaredType: "foo"},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
false,
|
|
|
|
},
|
|
|
|
|
|
|
|
// Output description
|
|
|
|
{
|
|
|
|
&Config{
|
|
|
|
Outputs: []*Output{
|
|
|
|
&Output{Description: "foo"},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
&Config{},
|
|
|
|
&Config{
|
|
|
|
Outputs: []*Output{
|
|
|
|
&Output{Description: "foo"},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
false,
|
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
&Config{},
|
|
|
|
&Config{
|
|
|
|
Outputs: []*Output{
|
|
|
|
&Output{Description: "foo"},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
&Config{
|
|
|
|
Outputs: []*Output{
|
|
|
|
&Output{Description: "foo"},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
false,
|
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
&Config{
|
|
|
|
Outputs: []*Output{
|
|
|
|
&Output{Description: "bar"},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
&Config{
|
|
|
|
Outputs: []*Output{
|
|
|
|
&Output{Description: "foo"},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
&Config{
|
|
|
|
Outputs: []*Output{
|
|
|
|
&Output{Description: "foo"},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
false,
|
|
|
|
},
|
|
|
|
|
|
|
|
// Output depends_on
|
|
|
|
{
|
|
|
|
&Config{
|
|
|
|
Outputs: []*Output{
|
|
|
|
&Output{DependsOn: []string{"foo"}},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
&Config{},
|
|
|
|
&Config{
|
|
|
|
Outputs: []*Output{
|
|
|
|
&Output{DependsOn: []string{"foo"}},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
false,
|
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
&Config{},
|
|
|
|
&Config{
|
|
|
|
Outputs: []*Output{
|
|
|
|
&Output{DependsOn: []string{"foo"}},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
&Config{
|
|
|
|
Outputs: []*Output{
|
|
|
|
&Output{DependsOn: []string{"foo"}},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
false,
|
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
&Config{
|
|
|
|
Outputs: []*Output{
|
|
|
|
&Output{DependsOn: []string{"bar"}},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
&Config{
|
|
|
|
Outputs: []*Output{
|
|
|
|
&Output{DependsOn: []string{"foo"}},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
&Config{
|
|
|
|
Outputs: []*Output{
|
|
|
|
&Output{DependsOn: []string{"foo"}},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
false,
|
|
|
|
},
|
|
|
|
|
|
|
|
// Output sensitive
|
|
|
|
{
|
|
|
|
&Config{
|
|
|
|
Outputs: []*Output{
|
|
|
|
&Output{Sensitive: true},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
&Config{},
|
|
|
|
&Config{
|
|
|
|
Outputs: []*Output{
|
|
|
|
&Output{Sensitive: true},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
false,
|
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
&Config{},
|
|
|
|
&Config{
|
|
|
|
Outputs: []*Output{
|
|
|
|
&Output{Sensitive: true},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
&Config{
|
|
|
|
Outputs: []*Output{
|
|
|
|
&Output{Sensitive: true},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
false,
|
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
&Config{
|
|
|
|
Outputs: []*Output{
|
|
|
|
&Output{Sensitive: false},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
&Config{
|
|
|
|
Outputs: []*Output{
|
|
|
|
&Output{Sensitive: true},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
&Config{
|
|
|
|
Outputs: []*Output{
|
|
|
|
&Output{Sensitive: true},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
false,
|
|
|
|
},
|
2014-07-21 02:17:03 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
for i, tc := range cases {
|
|
|
|
actual, err := Merge(tc.c1, tc.c2)
|
2015-10-08 14:48:04 +02:00
|
|
|
if err != nil != tc.err {
|
2014-07-21 02:17:03 +02:00
|
|
|
t.Fatalf("%d: error fail", i)
|
|
|
|
}
|
|
|
|
|
|
|
|
if !reflect.DeepEqual(actual, tc.result) {
|
|
|
|
t.Fatalf("%d: bad:\n\n%#v", i, actual)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|