2014-08-16 01:32:43 +02:00
|
|
|
package schema
|
|
|
|
|
|
|
|
import (
|
|
|
|
"reflect"
|
|
|
|
"testing"
|
|
|
|
|
|
|
|
"github.com/hashicorp/terraform/terraform"
|
|
|
|
)
|
|
|
|
|
|
|
|
func TestResourceDataGet(t *testing.T) {
|
|
|
|
cases := []struct {
|
|
|
|
Schema map[string]*Schema
|
|
|
|
State *terraform.ResourceState
|
|
|
|
Diff *terraform.ResourceDiff
|
|
|
|
Key string
|
|
|
|
Value interface{}
|
|
|
|
}{
|
2014-08-19 00:10:53 +02:00
|
|
|
{
|
|
|
|
Schema: map[string]*Schema{
|
|
|
|
"availability_zone": &Schema{
|
|
|
|
Type: TypeString,
|
|
|
|
Optional: true,
|
|
|
|
Computed: true,
|
|
|
|
ForceNew: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
State: nil,
|
|
|
|
|
|
|
|
Diff: &terraform.ResourceDiff{
|
|
|
|
Attributes: map[string]*terraform.ResourceAttrDiff{
|
|
|
|
"availability_zone": &terraform.ResourceAttrDiff{
|
|
|
|
Old: "",
|
|
|
|
New: "",
|
|
|
|
NewComputed: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
2014-08-22 08:03:04 +02:00
|
|
|
Key: "availability_zone",
|
|
|
|
Value: "",
|
2014-08-19 00:10:53 +02:00
|
|
|
},
|
|
|
|
|
2014-08-16 01:32:43 +02:00
|
|
|
{
|
|
|
|
Schema: map[string]*Schema{
|
|
|
|
"availability_zone": &Schema{
|
|
|
|
Type: TypeString,
|
|
|
|
Optional: true,
|
|
|
|
Computed: true,
|
|
|
|
ForceNew: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
State: nil,
|
|
|
|
|
|
|
|
Diff: &terraform.ResourceDiff{
|
|
|
|
Attributes: map[string]*terraform.ResourceAttrDiff{
|
|
|
|
"availability_zone": &terraform.ResourceAttrDiff{
|
|
|
|
Old: "",
|
|
|
|
New: "foo",
|
|
|
|
RequiresNew: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
Key: "availability_zone",
|
|
|
|
|
|
|
|
Value: "foo",
|
|
|
|
},
|
|
|
|
|
2014-08-22 17:57:44 +02:00
|
|
|
{
|
|
|
|
Schema: map[string]*Schema{
|
|
|
|
"availability_zone": &Schema{
|
|
|
|
Type: TypeString,
|
|
|
|
Optional: true,
|
|
|
|
Computed: true,
|
|
|
|
ForceNew: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
State: nil,
|
|
|
|
|
|
|
|
Diff: &terraform.ResourceDiff{
|
|
|
|
Attributes: map[string]*terraform.ResourceAttrDiff{
|
|
|
|
"availability_zone": &terraform.ResourceAttrDiff{
|
2014-08-22 21:09:06 +02:00
|
|
|
Old: "",
|
|
|
|
New: "foo!",
|
|
|
|
NewExtra: "foo",
|
2014-08-22 17:57:44 +02:00
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
Key: "availability_zone",
|
|
|
|
Value: "foo",
|
|
|
|
},
|
|
|
|
|
2014-08-16 01:32:43 +02:00
|
|
|
{
|
|
|
|
Schema: map[string]*Schema{
|
|
|
|
"availability_zone": &Schema{
|
|
|
|
Type: TypeString,
|
|
|
|
Optional: true,
|
|
|
|
Computed: true,
|
|
|
|
ForceNew: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
State: &terraform.ResourceState{
|
|
|
|
Attributes: map[string]string{
|
|
|
|
"availability_zone": "bar",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
Diff: nil,
|
|
|
|
|
|
|
|
Key: "availability_zone",
|
|
|
|
|
|
|
|
Value: "bar",
|
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
Schema: map[string]*Schema{
|
|
|
|
"port": &Schema{
|
|
|
|
Type: TypeInt,
|
|
|
|
Optional: true,
|
|
|
|
Computed: true,
|
|
|
|
ForceNew: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
State: &terraform.ResourceState{
|
|
|
|
Attributes: map[string]string{
|
|
|
|
"port": "80",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
Diff: nil,
|
|
|
|
|
|
|
|
Key: "port",
|
|
|
|
|
|
|
|
Value: 80,
|
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
Schema: map[string]*Schema{
|
|
|
|
"ports": &Schema{
|
|
|
|
Type: TypeList,
|
|
|
|
Required: true,
|
|
|
|
Elem: &Schema{Type: TypeInt},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
State: &terraform.ResourceState{
|
|
|
|
Attributes: map[string]string{
|
|
|
|
"ports.#": "3",
|
|
|
|
"ports.0": "1",
|
|
|
|
"ports.1": "2",
|
|
|
|
"ports.2": "5",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
Key: "ports.1",
|
|
|
|
|
|
|
|
Value: 2,
|
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
Schema: map[string]*Schema{
|
|
|
|
"ports": &Schema{
|
|
|
|
Type: TypeList,
|
|
|
|
Required: true,
|
|
|
|
Elem: &Schema{Type: TypeInt},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
State: &terraform.ResourceState{
|
|
|
|
Attributes: map[string]string{
|
|
|
|
"ports.#": "3",
|
|
|
|
"ports.0": "1",
|
|
|
|
"ports.1": "2",
|
|
|
|
"ports.2": "5",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
Key: "ports.#",
|
|
|
|
|
|
|
|
Value: 3,
|
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
Schema: map[string]*Schema{
|
|
|
|
"ports": &Schema{
|
|
|
|
Type: TypeList,
|
|
|
|
Required: true,
|
|
|
|
Elem: &Schema{Type: TypeInt},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
State: nil,
|
|
|
|
|
|
|
|
Key: "ports.#",
|
|
|
|
|
|
|
|
Value: 0,
|
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
Schema: map[string]*Schema{
|
|
|
|
"ports": &Schema{
|
|
|
|
Type: TypeList,
|
|
|
|
Required: true,
|
|
|
|
Elem: &Schema{Type: TypeInt},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
State: &terraform.ResourceState{
|
|
|
|
Attributes: map[string]string{
|
|
|
|
"ports.#": "3",
|
|
|
|
"ports.0": "1",
|
|
|
|
"ports.1": "2",
|
|
|
|
"ports.2": "5",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
Key: "ports",
|
|
|
|
|
|
|
|
Value: []interface{}{1, 2, 5},
|
|
|
|
},
|
2014-08-16 02:39:08 +02:00
|
|
|
|
|
|
|
{
|
|
|
|
Schema: map[string]*Schema{
|
|
|
|
"ingress": &Schema{
|
|
|
|
Type: TypeList,
|
|
|
|
Required: true,
|
|
|
|
Elem: &Resource{
|
|
|
|
Schema: map[string]*Schema{
|
|
|
|
"from": &Schema{
|
|
|
|
Type: TypeInt,
|
|
|
|
Required: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
State: nil,
|
|
|
|
|
|
|
|
Diff: &terraform.ResourceDiff{
|
|
|
|
Attributes: map[string]*terraform.ResourceAttrDiff{
|
|
|
|
"ingress.#": &terraform.ResourceAttrDiff{
|
|
|
|
Old: "",
|
|
|
|
New: "1",
|
|
|
|
},
|
|
|
|
"ingress.0.from": &terraform.ResourceAttrDiff{
|
|
|
|
Old: "",
|
|
|
|
New: "8080",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
Key: "ingress.0",
|
|
|
|
|
|
|
|
Value: map[string]interface{}{
|
|
|
|
"from": 8080,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
Schema: map[string]*Schema{
|
|
|
|
"ingress": &Schema{
|
|
|
|
Type: TypeList,
|
|
|
|
Required: true,
|
|
|
|
Elem: &Resource{
|
|
|
|
Schema: map[string]*Schema{
|
|
|
|
"from": &Schema{
|
|
|
|
Type: TypeInt,
|
|
|
|
Required: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
State: nil,
|
|
|
|
|
|
|
|
Diff: &terraform.ResourceDiff{
|
|
|
|
Attributes: map[string]*terraform.ResourceAttrDiff{
|
|
|
|
"ingress.#": &terraform.ResourceAttrDiff{
|
|
|
|
Old: "",
|
|
|
|
New: "1",
|
|
|
|
},
|
|
|
|
"ingress.0.from": &terraform.ResourceAttrDiff{
|
|
|
|
Old: "",
|
|
|
|
New: "8080",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
Key: "ingress",
|
|
|
|
|
|
|
|
Value: []interface{}{
|
|
|
|
map[string]interface{}{
|
|
|
|
"from": 8080,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
2014-08-16 22:32:21 +02:00
|
|
|
|
|
|
|
// Computed get
|
|
|
|
{
|
|
|
|
Schema: map[string]*Schema{
|
|
|
|
"availability_zone": &Schema{
|
|
|
|
Type: TypeString,
|
|
|
|
Computed: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
State: &terraform.ResourceState{
|
|
|
|
Attributes: map[string]string{
|
|
|
|
"availability_zone": "foo",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
Key: "availability_zone",
|
|
|
|
|
|
|
|
Value: "foo",
|
|
|
|
},
|
2014-08-16 22:55:10 +02:00
|
|
|
|
|
|
|
// Full object
|
|
|
|
{
|
|
|
|
Schema: map[string]*Schema{
|
|
|
|
"availability_zone": &Schema{
|
|
|
|
Type: TypeString,
|
|
|
|
Optional: true,
|
|
|
|
Computed: true,
|
|
|
|
ForceNew: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
State: nil,
|
|
|
|
|
|
|
|
Diff: &terraform.ResourceDiff{
|
|
|
|
Attributes: map[string]*terraform.ResourceAttrDiff{
|
|
|
|
"availability_zone": &terraform.ResourceAttrDiff{
|
|
|
|
Old: "",
|
|
|
|
New: "foo",
|
|
|
|
RequiresNew: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
Key: "",
|
|
|
|
|
|
|
|
Value: map[string]interface{}{
|
|
|
|
"availability_zone": "foo",
|
|
|
|
},
|
|
|
|
},
|
2014-08-18 23:00:03 +02:00
|
|
|
|
|
|
|
// List of maps
|
|
|
|
{
|
|
|
|
Schema: map[string]*Schema{
|
|
|
|
"config_vars": &Schema{
|
|
|
|
Type: TypeList,
|
|
|
|
Optional: true,
|
|
|
|
Computed: true,
|
|
|
|
Elem: &Schema{
|
|
|
|
Type: TypeMap,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
State: nil,
|
|
|
|
|
|
|
|
Diff: &terraform.ResourceDiff{
|
|
|
|
Attributes: map[string]*terraform.ResourceAttrDiff{
|
|
|
|
"config_vars.#": &terraform.ResourceAttrDiff{
|
|
|
|
Old: "0",
|
|
|
|
New: "2",
|
|
|
|
},
|
|
|
|
"config_vars.0.foo": &terraform.ResourceAttrDiff{
|
|
|
|
Old: "",
|
|
|
|
New: "bar",
|
|
|
|
},
|
|
|
|
"config_vars.1.bar": &terraform.ResourceAttrDiff{
|
|
|
|
Old: "",
|
|
|
|
New: "baz",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
Key: "config_vars",
|
|
|
|
|
|
|
|
Value: []interface{}{
|
|
|
|
map[string]interface{}{
|
|
|
|
"foo": "bar",
|
|
|
|
},
|
|
|
|
map[string]interface{}{
|
|
|
|
"bar": "baz",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
// List of maps in state
|
|
|
|
{
|
|
|
|
Schema: map[string]*Schema{
|
|
|
|
"config_vars": &Schema{
|
|
|
|
Type: TypeList,
|
|
|
|
Optional: true,
|
|
|
|
Computed: true,
|
|
|
|
Elem: &Schema{
|
|
|
|
Type: TypeMap,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
State: &terraform.ResourceState{
|
|
|
|
Attributes: map[string]string{
|
|
|
|
"config_vars.#": "2",
|
|
|
|
"config_vars.0.foo": "baz",
|
|
|
|
"config_vars.1.bar": "bar",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
Diff: nil,
|
|
|
|
|
|
|
|
Key: "config_vars",
|
|
|
|
|
|
|
|
Value: []interface{}{
|
|
|
|
map[string]interface{}{
|
|
|
|
"foo": "baz",
|
|
|
|
},
|
|
|
|
map[string]interface{}{
|
|
|
|
"bar": "bar",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
2014-08-19 04:12:49 +02:00
|
|
|
|
|
|
|
// List of maps with removal in diff
|
|
|
|
{
|
|
|
|
Schema: map[string]*Schema{
|
|
|
|
"config_vars": &Schema{
|
|
|
|
Type: TypeList,
|
|
|
|
Optional: true,
|
|
|
|
Computed: true,
|
|
|
|
Elem: &Schema{
|
|
|
|
Type: TypeMap,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
State: &terraform.ResourceState{
|
|
|
|
Attributes: map[string]string{
|
|
|
|
"config_vars.#": "1",
|
|
|
|
"config_vars.0.FOO": "bar",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
Diff: &terraform.ResourceDiff{
|
|
|
|
Attributes: map[string]*terraform.ResourceAttrDiff{
|
|
|
|
"config_vars.#": &terraform.ResourceAttrDiff{
|
|
|
|
Old: "1",
|
|
|
|
New: "0",
|
|
|
|
},
|
|
|
|
"config_vars.0.FOO": &terraform.ResourceAttrDiff{
|
|
|
|
Old: "bar",
|
|
|
|
NewRemoved: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
Key: "config_vars",
|
|
|
|
|
|
|
|
Value: []interface{}{},
|
|
|
|
},
|
2014-08-21 02:51:27 +02:00
|
|
|
|
|
|
|
// Sets
|
|
|
|
{
|
|
|
|
Schema: map[string]*Schema{
|
|
|
|
"ports": &Schema{
|
|
|
|
Type: TypeSet,
|
|
|
|
Optional: true,
|
|
|
|
Computed: true,
|
|
|
|
Elem: &Schema{Type: TypeInt},
|
|
|
|
Set: func(a interface{}) int {
|
|
|
|
return a.(int)
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
State: &terraform.ResourceState{
|
|
|
|
Attributes: map[string]string{
|
|
|
|
"ports.#": "1",
|
|
|
|
"ports.0": "80",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
Diff: nil,
|
|
|
|
|
|
|
|
Key: "ports",
|
|
|
|
|
|
|
|
Value: []interface{}{80},
|
|
|
|
},
|
2014-08-16 01:32:43 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
for i, tc := range cases {
|
|
|
|
d, err := schemaMap(tc.Schema).Data(tc.State, tc.Diff)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
v := d.Get(tc.Key)
|
2014-08-21 02:51:27 +02:00
|
|
|
if s, ok := v.(*Set); ok {
|
|
|
|
v = s.List()
|
|
|
|
}
|
|
|
|
|
2014-08-16 01:32:43 +02:00
|
|
|
if !reflect.DeepEqual(v, tc.Value) {
|
|
|
|
t.Fatalf("Bad: %d\n\n%#v", i, v)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2014-08-17 00:02:51 +02:00
|
|
|
|
2014-08-18 18:58:44 +02:00
|
|
|
func TestResourceDataGetChange(t *testing.T) {
|
|
|
|
cases := []struct {
|
|
|
|
Schema map[string]*Schema
|
|
|
|
State *terraform.ResourceState
|
|
|
|
Diff *terraform.ResourceDiff
|
|
|
|
Key string
|
|
|
|
OldValue interface{}
|
|
|
|
NewValue interface{}
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
Schema: map[string]*Schema{
|
|
|
|
"availability_zone": &Schema{
|
|
|
|
Type: TypeString,
|
|
|
|
Optional: true,
|
|
|
|
Computed: true,
|
|
|
|
ForceNew: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
State: nil,
|
|
|
|
|
|
|
|
Diff: &terraform.ResourceDiff{
|
|
|
|
Attributes: map[string]*terraform.ResourceAttrDiff{
|
|
|
|
"availability_zone": &terraform.ResourceAttrDiff{
|
|
|
|
Old: "",
|
|
|
|
New: "foo",
|
|
|
|
RequiresNew: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
Key: "availability_zone",
|
|
|
|
|
2014-08-22 08:03:04 +02:00
|
|
|
OldValue: "",
|
2014-08-18 18:58:44 +02:00
|
|
|
NewValue: "foo",
|
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
Schema: map[string]*Schema{
|
|
|
|
"availability_zone": &Schema{
|
|
|
|
Type: TypeString,
|
|
|
|
Optional: true,
|
|
|
|
Computed: true,
|
|
|
|
ForceNew: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
State: &terraform.ResourceState{
|
|
|
|
Attributes: map[string]string{
|
|
|
|
"availability_zone": "foo",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
Diff: &terraform.ResourceDiff{
|
|
|
|
Attributes: map[string]*terraform.ResourceAttrDiff{
|
|
|
|
"availability_zone": &terraform.ResourceAttrDiff{
|
|
|
|
Old: "",
|
|
|
|
New: "foo",
|
|
|
|
RequiresNew: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
Key: "availability_zone",
|
|
|
|
|
|
|
|
OldValue: "foo",
|
|
|
|
NewValue: "foo",
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for i, tc := range cases {
|
|
|
|
d, err := schemaMap(tc.Schema).Data(tc.State, tc.Diff)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
o, n := d.GetChange(tc.Key)
|
|
|
|
if !reflect.DeepEqual(o, tc.OldValue) {
|
|
|
|
t.Fatalf("Old Bad: %d\n\n%#v", i, o)
|
|
|
|
}
|
|
|
|
if !reflect.DeepEqual(n, tc.NewValue) {
|
|
|
|
t.Fatalf("New Bad: %d\n\n%#v", i, n)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-08-22 08:03:04 +02:00
|
|
|
func TestResourceDataGetOk(t *testing.T) {
|
|
|
|
cases := []struct {
|
|
|
|
Schema map[string]*Schema
|
|
|
|
State *terraform.ResourceState
|
|
|
|
Diff *terraform.ResourceDiff
|
|
|
|
Key string
|
|
|
|
Value interface{}
|
|
|
|
Ok bool
|
|
|
|
}{
|
|
|
|
/*
|
|
|
|
* Primitives
|
|
|
|
*/
|
|
|
|
{
|
|
|
|
Schema: map[string]*Schema{
|
|
|
|
"availability_zone": &Schema{
|
|
|
|
Type: TypeString,
|
|
|
|
Optional: true,
|
|
|
|
Computed: true,
|
|
|
|
ForceNew: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
State: nil,
|
|
|
|
|
|
|
|
Diff: &terraform.ResourceDiff{
|
|
|
|
Attributes: map[string]*terraform.ResourceAttrDiff{
|
|
|
|
"availability_zone": &terraform.ResourceAttrDiff{
|
|
|
|
Old: "",
|
|
|
|
New: "",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
Key: "availability_zone",
|
|
|
|
Value: "",
|
|
|
|
Ok: true,
|
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
Schema: map[string]*Schema{
|
|
|
|
"availability_zone": &Schema{
|
|
|
|
Type: TypeString,
|
|
|
|
Optional: true,
|
|
|
|
Computed: true,
|
|
|
|
ForceNew: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
State: nil,
|
|
|
|
|
|
|
|
Diff: nil,
|
|
|
|
|
|
|
|
Key: "availability_zone",
|
|
|
|
Value: "",
|
|
|
|
Ok: false,
|
|
|
|
},
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Lists
|
|
|
|
*/
|
|
|
|
|
|
|
|
{
|
|
|
|
Schema: map[string]*Schema{
|
|
|
|
"ports": &Schema{
|
|
|
|
Type: TypeList,
|
|
|
|
Optional: true,
|
|
|
|
Elem: &Schema{Type: TypeInt},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
State: nil,
|
|
|
|
|
|
|
|
Diff: nil,
|
|
|
|
|
|
|
|
Key: "ports",
|
|
|
|
Value: []interface{}{},
|
|
|
|
Ok: false,
|
|
|
|
},
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Map
|
|
|
|
*/
|
|
|
|
|
|
|
|
{
|
|
|
|
Schema: map[string]*Schema{
|
|
|
|
"ports": &Schema{
|
|
|
|
Type: TypeMap,
|
|
|
|
Optional: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
State: nil,
|
|
|
|
|
|
|
|
Diff: nil,
|
|
|
|
|
|
|
|
Key: "ports",
|
|
|
|
Value: map[string]interface{}{},
|
|
|
|
Ok: false,
|
|
|
|
},
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Set
|
|
|
|
*/
|
|
|
|
|
|
|
|
{
|
|
|
|
Schema: map[string]*Schema{
|
|
|
|
"ports": &Schema{
|
|
|
|
Type: TypeSet,
|
|
|
|
Optional: true,
|
|
|
|
Elem: &Schema{Type: TypeInt},
|
|
|
|
Set: func(a interface{}) int { return a.(int) },
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
State: nil,
|
|
|
|
|
|
|
|
Diff: nil,
|
|
|
|
|
|
|
|
Key: "ports",
|
|
|
|
Value: []interface{}{},
|
|
|
|
Ok: false,
|
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
Schema: map[string]*Schema{
|
|
|
|
"ports": &Schema{
|
|
|
|
Type: TypeSet,
|
|
|
|
Optional: true,
|
|
|
|
Elem: &Schema{Type: TypeInt},
|
|
|
|
Set: func(a interface{}) int { return a.(int) },
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
State: nil,
|
|
|
|
|
|
|
|
Diff: nil,
|
|
|
|
|
|
|
|
Key: "ports.0",
|
|
|
|
Value: 0,
|
|
|
|
Ok: false,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for i, tc := range cases {
|
|
|
|
d, err := schemaMap(tc.Schema).Data(tc.State, tc.Diff)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
v, ok := d.GetOk(tc.Key)
|
|
|
|
if s, ok := v.(*Set); ok {
|
|
|
|
v = s.List()
|
|
|
|
}
|
|
|
|
|
|
|
|
if !reflect.DeepEqual(v, tc.Value) {
|
|
|
|
t.Fatalf("Bad: %d\n\n%#v", i, v)
|
|
|
|
}
|
|
|
|
if ok != tc.Ok {
|
|
|
|
t.Fatalf("Bad: %d\n\n%#v", i, ok)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-08-18 19:00:41 +02:00
|
|
|
func TestResourceDataHasChange(t *testing.T) {
|
|
|
|
cases := []struct {
|
|
|
|
Schema map[string]*Schema
|
|
|
|
State *terraform.ResourceState
|
|
|
|
Diff *terraform.ResourceDiff
|
|
|
|
Key string
|
|
|
|
Change bool
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
Schema: map[string]*Schema{
|
|
|
|
"availability_zone": &Schema{
|
|
|
|
Type: TypeString,
|
|
|
|
Optional: true,
|
|
|
|
Computed: true,
|
|
|
|
ForceNew: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
State: nil,
|
|
|
|
|
|
|
|
Diff: &terraform.ResourceDiff{
|
|
|
|
Attributes: map[string]*terraform.ResourceAttrDiff{
|
|
|
|
"availability_zone": &terraform.ResourceAttrDiff{
|
|
|
|
Old: "",
|
|
|
|
New: "foo",
|
|
|
|
RequiresNew: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
Key: "availability_zone",
|
|
|
|
|
|
|
|
Change: true,
|
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
Schema: map[string]*Schema{
|
|
|
|
"availability_zone": &Schema{
|
|
|
|
Type: TypeString,
|
|
|
|
Optional: true,
|
|
|
|
Computed: true,
|
|
|
|
ForceNew: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
State: &terraform.ResourceState{
|
|
|
|
Attributes: map[string]string{
|
|
|
|
"availability_zone": "foo",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
Diff: &terraform.ResourceDiff{
|
|
|
|
Attributes: map[string]*terraform.ResourceAttrDiff{
|
|
|
|
"availability_zone": &terraform.ResourceAttrDiff{
|
|
|
|
Old: "",
|
|
|
|
New: "foo",
|
|
|
|
RequiresNew: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
Key: "availability_zone",
|
|
|
|
|
|
|
|
Change: false,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for i, tc := range cases {
|
|
|
|
d, err := schemaMap(tc.Schema).Data(tc.State, tc.Diff)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
actual := d.HasChange(tc.Key)
|
|
|
|
if actual != tc.Change {
|
|
|
|
t.Fatalf("Bad: %d %#v", i, actual)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-08-17 00:02:51 +02:00
|
|
|
func TestResourceDataSet(t *testing.T) {
|
|
|
|
cases := []struct {
|
|
|
|
Schema map[string]*Schema
|
|
|
|
State *terraform.ResourceState
|
|
|
|
Diff *terraform.ResourceDiff
|
|
|
|
Key string
|
|
|
|
Value interface{}
|
|
|
|
Err bool
|
|
|
|
GetKey string
|
|
|
|
GetValue interface{}
|
|
|
|
}{
|
|
|
|
// Basic good
|
|
|
|
{
|
|
|
|
Schema: map[string]*Schema{
|
|
|
|
"availability_zone": &Schema{
|
|
|
|
Type: TypeString,
|
|
|
|
Optional: true,
|
|
|
|
Computed: true,
|
|
|
|
ForceNew: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
State: nil,
|
|
|
|
|
|
|
|
Diff: nil,
|
|
|
|
|
|
|
|
Key: "availability_zone",
|
|
|
|
Value: "foo",
|
|
|
|
|
|
|
|
GetKey: "availability_zone",
|
|
|
|
GetValue: "foo",
|
|
|
|
},
|
|
|
|
|
|
|
|
// Basic int
|
|
|
|
{
|
|
|
|
Schema: map[string]*Schema{
|
|
|
|
"port": &Schema{
|
|
|
|
Type: TypeInt,
|
|
|
|
Optional: true,
|
|
|
|
Computed: true,
|
|
|
|
ForceNew: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
State: nil,
|
|
|
|
|
|
|
|
Diff: nil,
|
|
|
|
|
|
|
|
Key: "port",
|
|
|
|
Value: 80,
|
|
|
|
|
|
|
|
GetKey: "port",
|
|
|
|
GetValue: 80,
|
|
|
|
},
|
|
|
|
|
2014-08-20 01:46:36 +02:00
|
|
|
// Basic bool
|
|
|
|
{
|
|
|
|
Schema: map[string]*Schema{
|
|
|
|
"vpc": &Schema{
|
|
|
|
Type: TypeBool,
|
|
|
|
Optional: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
State: nil,
|
|
|
|
|
|
|
|
Diff: nil,
|
|
|
|
|
|
|
|
Key: "vpc",
|
|
|
|
Value: true,
|
|
|
|
|
|
|
|
GetKey: "vpc",
|
|
|
|
GetValue: true,
|
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
Schema: map[string]*Schema{
|
|
|
|
"vpc": &Schema{
|
|
|
|
Type: TypeBool,
|
|
|
|
Optional: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
State: nil,
|
|
|
|
|
|
|
|
Diff: nil,
|
|
|
|
|
|
|
|
Key: "vpc",
|
|
|
|
Value: false,
|
|
|
|
|
|
|
|
GetKey: "vpc",
|
|
|
|
GetValue: false,
|
|
|
|
},
|
|
|
|
|
2014-08-17 00:02:51 +02:00
|
|
|
// Invalid type
|
|
|
|
{
|
|
|
|
Schema: map[string]*Schema{
|
|
|
|
"availability_zone": &Schema{
|
|
|
|
Type: TypeString,
|
|
|
|
Optional: true,
|
|
|
|
Computed: true,
|
|
|
|
ForceNew: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
State: nil,
|
|
|
|
|
|
|
|
Diff: nil,
|
|
|
|
|
|
|
|
Key: "availability_zone",
|
|
|
|
Value: 80,
|
|
|
|
Err: true,
|
|
|
|
|
|
|
|
GetKey: "availability_zone",
|
2014-08-22 08:03:04 +02:00
|
|
|
GetValue: "",
|
2014-08-17 00:02:51 +02:00
|
|
|
},
|
2014-08-17 20:38:16 +02:00
|
|
|
|
|
|
|
// List of primitives, set element
|
|
|
|
{
|
|
|
|
Schema: map[string]*Schema{
|
|
|
|
"ports": &Schema{
|
|
|
|
Type: TypeList,
|
|
|
|
Computed: true,
|
|
|
|
Elem: &Schema{Type: TypeInt},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
State: &terraform.ResourceState{
|
|
|
|
Attributes: map[string]string{
|
|
|
|
"ports.#": "3",
|
|
|
|
"ports.0": "1",
|
|
|
|
"ports.1": "2",
|
|
|
|
"ports.2": "5",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
Diff: nil,
|
|
|
|
|
|
|
|
Key: "ports.1",
|
|
|
|
Value: 3,
|
|
|
|
|
|
|
|
GetKey: "ports",
|
|
|
|
GetValue: []interface{}{1, 3, 5},
|
|
|
|
},
|
|
|
|
|
|
|
|
// List of primitives, set list
|
|
|
|
{
|
|
|
|
Schema: map[string]*Schema{
|
|
|
|
"ports": &Schema{
|
|
|
|
Type: TypeList,
|
|
|
|
Computed: true,
|
|
|
|
Elem: &Schema{Type: TypeInt},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
State: nil,
|
|
|
|
|
|
|
|
Diff: nil,
|
|
|
|
|
|
|
|
Key: "ports",
|
|
|
|
Value: []int{1, 2, 5},
|
|
|
|
|
|
|
|
GetKey: "ports",
|
|
|
|
GetValue: []interface{}{1, 2, 5},
|
|
|
|
},
|
|
|
|
|
|
|
|
// List of primitives, set list with error
|
|
|
|
{
|
|
|
|
Schema: map[string]*Schema{
|
|
|
|
"ports": &Schema{
|
|
|
|
Type: TypeList,
|
|
|
|
Computed: true,
|
|
|
|
Elem: &Schema{Type: TypeInt},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
State: nil,
|
|
|
|
|
|
|
|
Diff: nil,
|
|
|
|
|
|
|
|
Key: "ports",
|
|
|
|
Value: []interface{}{1, "NOPE", 5},
|
|
|
|
Err: true,
|
|
|
|
|
|
|
|
GetKey: "ports",
|
|
|
|
GetValue: []interface{}{},
|
|
|
|
},
|
|
|
|
|
|
|
|
// List of resource, set element
|
|
|
|
{
|
|
|
|
Schema: map[string]*Schema{
|
|
|
|
"ingress": &Schema{
|
|
|
|
Type: TypeList,
|
|
|
|
Computed: true,
|
|
|
|
Elem: &Resource{
|
|
|
|
Schema: map[string]*Schema{
|
|
|
|
"from": &Schema{
|
|
|
|
Type: TypeInt,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
State: &terraform.ResourceState{
|
|
|
|
Attributes: map[string]string{
|
|
|
|
"ingress.#": "2",
|
|
|
|
"ingress.0.from": "80",
|
|
|
|
"ingress.1.from": "8080",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
Diff: nil,
|
|
|
|
|
|
|
|
Key: "ingress.1.from",
|
|
|
|
Value: 9000,
|
|
|
|
|
|
|
|
GetKey: "ingress",
|
|
|
|
GetValue: []interface{}{
|
|
|
|
map[string]interface{}{
|
|
|
|
"from": 80,
|
|
|
|
},
|
|
|
|
map[string]interface{}{
|
|
|
|
"from": 9000,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
// List of resource, set full resource element
|
|
|
|
{
|
|
|
|
Schema: map[string]*Schema{
|
|
|
|
"ingress": &Schema{
|
|
|
|
Type: TypeList,
|
|
|
|
Computed: true,
|
|
|
|
Elem: &Resource{
|
|
|
|
Schema: map[string]*Schema{
|
|
|
|
"from": &Schema{
|
|
|
|
Type: TypeInt,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
State: &terraform.ResourceState{
|
|
|
|
Attributes: map[string]string{
|
|
|
|
"ingress.#": "2",
|
|
|
|
"ingress.0.from": "80",
|
|
|
|
"ingress.1.from": "8080",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
Diff: nil,
|
|
|
|
|
|
|
|
Key: "ingress.1",
|
|
|
|
Value: map[string]interface{}{
|
|
|
|
"from": 9000,
|
|
|
|
},
|
|
|
|
|
|
|
|
GetKey: "ingress",
|
|
|
|
GetValue: []interface{}{
|
|
|
|
map[string]interface{}{
|
|
|
|
"from": 80,
|
|
|
|
},
|
|
|
|
map[string]interface{}{
|
|
|
|
"from": 9000,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
// List of resource, set full resource element, with error
|
|
|
|
{
|
|
|
|
Schema: map[string]*Schema{
|
|
|
|
"ingress": &Schema{
|
|
|
|
Type: TypeList,
|
|
|
|
Computed: true,
|
|
|
|
Elem: &Resource{
|
|
|
|
Schema: map[string]*Schema{
|
|
|
|
"from": &Schema{
|
|
|
|
Type: TypeInt,
|
|
|
|
},
|
|
|
|
"to": &Schema{
|
|
|
|
Type: TypeInt,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
State: &terraform.ResourceState{
|
|
|
|
Attributes: map[string]string{
|
|
|
|
"ingress.#": "2",
|
|
|
|
"ingress.0.from": "80",
|
|
|
|
"ingress.0.to": "10",
|
|
|
|
"ingress.1.from": "8080",
|
|
|
|
"ingress.1.to": "8080",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
Diff: nil,
|
|
|
|
|
|
|
|
Key: "ingress.1",
|
|
|
|
Value: map[string]interface{}{
|
|
|
|
"from": 9000,
|
|
|
|
"to": "bar",
|
|
|
|
},
|
|
|
|
Err: true,
|
|
|
|
|
|
|
|
GetKey: "ingress",
|
|
|
|
GetValue: []interface{}{
|
|
|
|
map[string]interface{}{
|
|
|
|
"from": 80,
|
|
|
|
"to": 10,
|
|
|
|
},
|
|
|
|
map[string]interface{}{
|
|
|
|
"from": 8080,
|
|
|
|
"to": 8080,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
2014-08-18 23:00:03 +02:00
|
|
|
|
|
|
|
// Set a list of maps
|
|
|
|
{
|
|
|
|
Schema: map[string]*Schema{
|
|
|
|
"config_vars": &Schema{
|
|
|
|
Type: TypeList,
|
|
|
|
Optional: true,
|
|
|
|
Computed: true,
|
|
|
|
Elem: &Schema{
|
|
|
|
Type: TypeMap,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
State: nil,
|
|
|
|
|
|
|
|
Diff: nil,
|
|
|
|
|
|
|
|
Key: "config_vars",
|
|
|
|
Value: []interface{}{
|
|
|
|
map[string]interface{}{
|
|
|
|
"foo": "bar",
|
|
|
|
},
|
|
|
|
map[string]interface{}{
|
|
|
|
"bar": "baz",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
Err: false,
|
|
|
|
|
|
|
|
GetKey: "config_vars",
|
|
|
|
GetValue: []interface{}{
|
|
|
|
map[string]interface{}{
|
|
|
|
"foo": "bar",
|
|
|
|
},
|
|
|
|
map[string]interface{}{
|
|
|
|
"bar": "baz",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
2014-08-19 00:17:18 +02:00
|
|
|
|
|
|
|
// Set a list of maps
|
|
|
|
{
|
|
|
|
Schema: map[string]*Schema{
|
|
|
|
"config_vars": &Schema{
|
|
|
|
Type: TypeList,
|
|
|
|
Optional: true,
|
|
|
|
Computed: true,
|
|
|
|
Elem: &Schema{
|
|
|
|
Type: TypeMap,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
State: nil,
|
|
|
|
|
|
|
|
Diff: nil,
|
|
|
|
|
|
|
|
Key: "config_vars",
|
|
|
|
Value: []interface{}{
|
|
|
|
map[string]string{
|
|
|
|
"foo": "bar",
|
|
|
|
},
|
|
|
|
map[string]string{
|
|
|
|
"bar": "baz",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
Err: false,
|
|
|
|
|
|
|
|
GetKey: "config_vars",
|
|
|
|
GetValue: []interface{}{
|
|
|
|
map[string]interface{}{
|
|
|
|
"foo": "bar",
|
|
|
|
},
|
|
|
|
map[string]interface{}{
|
|
|
|
"bar": "baz",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
2014-08-21 03:11:14 +02:00
|
|
|
|
|
|
|
// Set, with list
|
|
|
|
{
|
|
|
|
Schema: map[string]*Schema{
|
|
|
|
"ports": &Schema{
|
|
|
|
Type: TypeSet,
|
|
|
|
Optional: true,
|
|
|
|
Computed: true,
|
|
|
|
Elem: &Schema{Type: TypeInt},
|
|
|
|
Set: func(a interface{}) int {
|
|
|
|
return a.(int)
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
State: &terraform.ResourceState{
|
|
|
|
Attributes: map[string]string{
|
|
|
|
"ports.#": "3",
|
|
|
|
"ports.0": "100",
|
|
|
|
"ports.1": "80",
|
|
|
|
"ports.2": "80",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
Key: "ports",
|
|
|
|
Value: []interface{}{100, 125, 125},
|
|
|
|
|
|
|
|
GetKey: "ports",
|
|
|
|
GetValue: []interface{}{100, 125},
|
|
|
|
},
|
|
|
|
|
|
|
|
// Set, with Set
|
|
|
|
{
|
|
|
|
Schema: map[string]*Schema{
|
|
|
|
"ports": &Schema{
|
|
|
|
Type: TypeSet,
|
|
|
|
Optional: true,
|
|
|
|
Computed: true,
|
|
|
|
Elem: &Schema{Type: TypeInt},
|
|
|
|
Set: func(a interface{}) int {
|
|
|
|
return a.(int)
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
State: &terraform.ResourceState{
|
|
|
|
Attributes: map[string]string{
|
|
|
|
"ports.#": "3",
|
|
|
|
"ports.0": "100",
|
|
|
|
"ports.1": "80",
|
|
|
|
"ports.2": "80",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
Key: "ports",
|
|
|
|
Value: &Set{
|
|
|
|
m: map[int]interface{}{
|
|
|
|
1: 1,
|
|
|
|
2: 2,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
GetKey: "ports",
|
|
|
|
GetValue: []interface{}{1, 2},
|
|
|
|
},
|
|
|
|
|
|
|
|
// Set single item
|
|
|
|
{
|
|
|
|
Schema: map[string]*Schema{
|
|
|
|
"ports": &Schema{
|
|
|
|
Type: TypeSet,
|
|
|
|
Optional: true,
|
|
|
|
Computed: true,
|
|
|
|
Elem: &Schema{Type: TypeInt},
|
|
|
|
Set: func(a interface{}) int {
|
|
|
|
return a.(int)
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
State: &terraform.ResourceState{
|
|
|
|
Attributes: map[string]string{
|
|
|
|
"ports.#": "2",
|
|
|
|
"ports.0": "100",
|
|
|
|
"ports.1": "80",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
Key: "ports.0",
|
|
|
|
Value: 256,
|
|
|
|
Err: true,
|
|
|
|
|
|
|
|
GetKey: "ports",
|
|
|
|
GetValue: []interface{}{80, 100},
|
|
|
|
},
|
2014-08-17 00:02:51 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
for i, tc := range cases {
|
|
|
|
d, err := schemaMap(tc.Schema).Data(tc.State, tc.Diff)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
err = d.Set(tc.Key, tc.Value)
|
|
|
|
if (err != nil) != tc.Err {
|
|
|
|
t.Fatalf("%d err: %s", i, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
v := d.Get(tc.GetKey)
|
2014-08-21 03:11:14 +02:00
|
|
|
if s, ok := v.(*Set); ok {
|
|
|
|
v = s.List()
|
|
|
|
}
|
2014-08-17 00:02:51 +02:00
|
|
|
if !reflect.DeepEqual(v, tc.GetValue) {
|
|
|
|
t.Fatalf("Get Bad: %d\n\n%#v", i, v)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2014-08-17 23:12:54 +02:00
|
|
|
|
|
|
|
func TestResourceDataState(t *testing.T) {
|
|
|
|
cases := []struct {
|
|
|
|
Schema map[string]*Schema
|
|
|
|
State *terraform.ResourceState
|
|
|
|
Diff *terraform.ResourceDiff
|
|
|
|
Set map[string]interface{}
|
|
|
|
Result *terraform.ResourceState
|
|
|
|
}{
|
|
|
|
// Basic primitive in diff
|
|
|
|
{
|
|
|
|
Schema: map[string]*Schema{
|
|
|
|
"availability_zone": &Schema{
|
|
|
|
Type: TypeString,
|
|
|
|
Optional: true,
|
|
|
|
Computed: true,
|
|
|
|
ForceNew: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
State: nil,
|
|
|
|
|
|
|
|
Diff: &terraform.ResourceDiff{
|
|
|
|
Attributes: map[string]*terraform.ResourceAttrDiff{
|
|
|
|
"availability_zone": &terraform.ResourceAttrDiff{
|
|
|
|
Old: "",
|
|
|
|
New: "foo",
|
|
|
|
RequiresNew: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
Result: &terraform.ResourceState{
|
|
|
|
Attributes: map[string]string{
|
|
|
|
"availability_zone": "foo",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
// Basic primitive set override
|
|
|
|
{
|
|
|
|
Schema: map[string]*Schema{
|
|
|
|
"availability_zone": &Schema{
|
|
|
|
Type: TypeString,
|
|
|
|
Optional: true,
|
|
|
|
Computed: true,
|
|
|
|
ForceNew: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
State: nil,
|
|
|
|
|
|
|
|
Diff: &terraform.ResourceDiff{
|
|
|
|
Attributes: map[string]*terraform.ResourceAttrDiff{
|
|
|
|
"availability_zone": &terraform.ResourceAttrDiff{
|
|
|
|
Old: "",
|
|
|
|
New: "foo",
|
|
|
|
RequiresNew: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
Set: map[string]interface{}{
|
|
|
|
"availability_zone": "bar",
|
|
|
|
},
|
|
|
|
|
|
|
|
Result: &terraform.ResourceState{
|
|
|
|
Attributes: map[string]string{
|
|
|
|
"availability_zone": "bar",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
2014-08-20 01:46:36 +02:00
|
|
|
{
|
|
|
|
Schema: map[string]*Schema{
|
|
|
|
"vpc": &Schema{
|
|
|
|
Type: TypeBool,
|
|
|
|
Optional: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
State: nil,
|
|
|
|
|
|
|
|
Diff: nil,
|
|
|
|
|
|
|
|
Set: map[string]interface{}{
|
|
|
|
"vpc": true,
|
|
|
|
},
|
|
|
|
|
|
|
|
Result: &terraform.ResourceState{
|
|
|
|
Attributes: map[string]string{
|
|
|
|
"vpc": "true",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
2014-08-22 17:45:54 +02:00
|
|
|
// Basic primitive with StateFunc set
|
|
|
|
{
|
|
|
|
Schema: map[string]*Schema{
|
|
|
|
"availability_zone": &Schema{
|
|
|
|
Type: TypeString,
|
|
|
|
Optional: true,
|
|
|
|
Computed: true,
|
|
|
|
StateFunc: func(interface{}) string { return "" },
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
State: nil,
|
|
|
|
|
|
|
|
Diff: &terraform.ResourceDiff{
|
|
|
|
Attributes: map[string]*terraform.ResourceAttrDiff{
|
|
|
|
"availability_zone": &terraform.ResourceAttrDiff{
|
2014-08-22 21:09:06 +02:00
|
|
|
Old: "",
|
|
|
|
New: "foo",
|
|
|
|
NewExtra: "foo!",
|
2014-08-22 17:45:54 +02:00
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
Result: &terraform.ResourceState{
|
|
|
|
Attributes: map[string]string{
|
|
|
|
"availability_zone": "foo",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
2014-08-17 23:12:54 +02:00
|
|
|
// List
|
|
|
|
{
|
|
|
|
Schema: map[string]*Schema{
|
|
|
|
"ports": &Schema{
|
|
|
|
Type: TypeList,
|
|
|
|
Required: true,
|
|
|
|
Elem: &Schema{Type: TypeInt},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
State: &terraform.ResourceState{
|
|
|
|
Attributes: map[string]string{
|
|
|
|
"ports.#": "1",
|
|
|
|
"ports.0": "80",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
Diff: &terraform.ResourceDiff{
|
|
|
|
Attributes: map[string]*terraform.ResourceAttrDiff{
|
|
|
|
"ports.#": &terraform.ResourceAttrDiff{
|
|
|
|
Old: "1",
|
|
|
|
New: "2",
|
|
|
|
},
|
|
|
|
"ports.1": &terraform.ResourceAttrDiff{
|
|
|
|
Old: "",
|
|
|
|
New: "100",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
Result: &terraform.ResourceState{
|
|
|
|
Attributes: map[string]string{
|
|
|
|
"ports.#": "2",
|
|
|
|
"ports.0": "80",
|
|
|
|
"ports.1": "100",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
// List of resources
|
|
|
|
{
|
|
|
|
Schema: map[string]*Schema{
|
|
|
|
"ingress": &Schema{
|
|
|
|
Type: TypeList,
|
|
|
|
Required: true,
|
|
|
|
Elem: &Resource{
|
|
|
|
Schema: map[string]*Schema{
|
|
|
|
"from": &Schema{
|
|
|
|
Type: TypeInt,
|
|
|
|
Required: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
State: &terraform.ResourceState{
|
|
|
|
Attributes: map[string]string{
|
|
|
|
"ingress.#": "1",
|
|
|
|
"ingress.0.from": "80",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
Diff: &terraform.ResourceDiff{
|
|
|
|
Attributes: map[string]*terraform.ResourceAttrDiff{
|
|
|
|
"ingress.#": &terraform.ResourceAttrDiff{
|
|
|
|
Old: "1",
|
|
|
|
New: "2",
|
|
|
|
},
|
|
|
|
"ingress.0.from": &terraform.ResourceAttrDiff{
|
|
|
|
Old: "80",
|
|
|
|
New: "150",
|
|
|
|
},
|
|
|
|
"ingress.1.from": &terraform.ResourceAttrDiff{
|
|
|
|
Old: "",
|
|
|
|
New: "100",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
Result: &terraform.ResourceState{
|
|
|
|
Attributes: map[string]string{
|
|
|
|
"ingress.#": "2",
|
|
|
|
"ingress.0.from": "150",
|
|
|
|
"ingress.1.from": "100",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
2014-08-18 23:00:03 +02:00
|
|
|
|
|
|
|
// List of maps
|
|
|
|
{
|
|
|
|
Schema: map[string]*Schema{
|
|
|
|
"config_vars": &Schema{
|
|
|
|
Type: TypeList,
|
|
|
|
Optional: true,
|
|
|
|
Computed: true,
|
|
|
|
Elem: &Schema{
|
|
|
|
Type: TypeMap,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
State: &terraform.ResourceState{
|
|
|
|
Attributes: map[string]string{
|
|
|
|
"config_vars.#": "2",
|
|
|
|
"config_vars.0.foo": "bar",
|
2014-08-19 00:07:09 +02:00
|
|
|
"config_vars.0.bar": "bar",
|
2014-08-18 23:00:03 +02:00
|
|
|
"config_vars.1.bar": "baz",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
2014-08-19 00:07:09 +02:00
|
|
|
Diff: &terraform.ResourceDiff{
|
|
|
|
Attributes: map[string]*terraform.ResourceAttrDiff{
|
|
|
|
"config_vars.0.bar": &terraform.ResourceAttrDiff{
|
|
|
|
NewRemoved: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
2014-08-18 23:00:03 +02:00
|
|
|
Set: map[string]interface{}{
|
|
|
|
"config_vars.1": map[string]interface{}{
|
|
|
|
"baz": "bang",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
Result: &terraform.ResourceState{
|
|
|
|
Attributes: map[string]string{
|
|
|
|
"config_vars.#": "2",
|
|
|
|
"config_vars.0.foo": "bar",
|
|
|
|
"config_vars.1.baz": "bang",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
2014-08-19 04:12:49 +02:00
|
|
|
|
|
|
|
// List of maps with removal in diff
|
|
|
|
{
|
|
|
|
Schema: map[string]*Schema{
|
|
|
|
"config_vars": &Schema{
|
|
|
|
Type: TypeList,
|
|
|
|
Optional: true,
|
|
|
|
Computed: true,
|
|
|
|
Elem: &Schema{
|
|
|
|
Type: TypeMap,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
State: &terraform.ResourceState{
|
|
|
|
Attributes: map[string]string{
|
|
|
|
"config_vars.#": "1",
|
|
|
|
"config_vars.0.FOO": "bar",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
Diff: &terraform.ResourceDiff{
|
|
|
|
Attributes: map[string]*terraform.ResourceAttrDiff{
|
|
|
|
"config_vars.#": &terraform.ResourceAttrDiff{
|
|
|
|
Old: "1",
|
|
|
|
New: "0",
|
|
|
|
},
|
|
|
|
"config_vars.0.FOO": &terraform.ResourceAttrDiff{
|
|
|
|
Old: "bar",
|
|
|
|
NewRemoved: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
Result: &terraform.ResourceState{
|
|
|
|
Attributes: map[string]string{},
|
|
|
|
},
|
|
|
|
},
|
2014-08-20 19:37:59 +02:00
|
|
|
|
|
|
|
// Basic state with other keys
|
|
|
|
{
|
|
|
|
Schema: map[string]*Schema{
|
|
|
|
"availability_zone": &Schema{
|
|
|
|
Type: TypeString,
|
|
|
|
Optional: true,
|
|
|
|
Computed: true,
|
|
|
|
ForceNew: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
State: &terraform.ResourceState{
|
|
|
|
ID: "bar",
|
|
|
|
Attributes: map[string]string{
|
|
|
|
"id": "bar",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
Diff: &terraform.ResourceDiff{
|
|
|
|
Attributes: map[string]*terraform.ResourceAttrDiff{
|
|
|
|
"availability_zone": &terraform.ResourceAttrDiff{
|
|
|
|
Old: "",
|
|
|
|
New: "foo",
|
|
|
|
RequiresNew: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
Result: &terraform.ResourceState{
|
|
|
|
ID: "bar",
|
|
|
|
Attributes: map[string]string{
|
|
|
|
"id": "bar",
|
|
|
|
"availability_zone": "foo",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
2014-08-21 02:51:27 +02:00
|
|
|
|
|
|
|
// Sets
|
|
|
|
{
|
|
|
|
Schema: map[string]*Schema{
|
|
|
|
"ports": &Schema{
|
|
|
|
Type: TypeSet,
|
|
|
|
Optional: true,
|
|
|
|
Computed: true,
|
|
|
|
Elem: &Schema{Type: TypeInt},
|
|
|
|
Set: func(a interface{}) int {
|
|
|
|
return a.(int)
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
State: &terraform.ResourceState{
|
|
|
|
Attributes: map[string]string{
|
2014-08-21 02:52:01 +02:00
|
|
|
"ports.#": "3",
|
2014-08-21 02:51:27 +02:00
|
|
|
"ports.0": "100",
|
|
|
|
"ports.1": "80",
|
2014-08-21 02:52:01 +02:00
|
|
|
"ports.2": "80",
|
2014-08-21 02:51:27 +02:00
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
Diff: nil,
|
|
|
|
|
|
|
|
Result: &terraform.ResourceState{
|
|
|
|
Attributes: map[string]string{
|
|
|
|
"ports.#": "2",
|
|
|
|
"ports.0": "80",
|
|
|
|
"ports.1": "100",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
2014-08-17 23:12:54 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
for i, tc := range cases {
|
|
|
|
d, err := schemaMap(tc.Schema).Data(tc.State, tc.Diff)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
for k, v := range tc.Set {
|
|
|
|
if err := d.Set(k, v); err != nil {
|
|
|
|
t.Fatalf("%d err: %s", i, err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-08-22 07:19:33 +02:00
|
|
|
// Set an ID so that the state returned is not nil
|
|
|
|
idSet := false
|
|
|
|
if d.Id() == "" {
|
|
|
|
idSet = true
|
|
|
|
d.SetId("foo")
|
|
|
|
}
|
|
|
|
|
2014-08-17 23:12:54 +02:00
|
|
|
actual := d.State()
|
2014-08-22 07:19:33 +02:00
|
|
|
|
|
|
|
// If we set an ID, then undo what we did so the comparison works
|
|
|
|
if actual != nil && idSet {
|
|
|
|
actual.ID = ""
|
|
|
|
delete(actual.Attributes, "id")
|
|
|
|
}
|
|
|
|
|
2014-08-17 23:12:54 +02:00
|
|
|
if !reflect.DeepEqual(actual, tc.Result) {
|
|
|
|
t.Fatalf("Bad: %d\n\n%#v", i, actual)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2014-08-18 05:48:50 +02:00
|
|
|
|
2014-08-22 07:15:47 +02:00
|
|
|
func TestResourceDataSetConnInfo(t *testing.T) {
|
|
|
|
d := &ResourceData{}
|
2014-08-22 07:19:33 +02:00
|
|
|
d.SetId("foo")
|
2014-08-22 07:15:47 +02:00
|
|
|
d.SetConnInfo(map[string]string{
|
|
|
|
"foo": "bar",
|
|
|
|
})
|
|
|
|
|
|
|
|
expected := map[string]string{
|
|
|
|
"foo": "bar",
|
|
|
|
}
|
|
|
|
|
|
|
|
actual := d.State()
|
|
|
|
if !reflect.DeepEqual(actual.ConnInfo, expected) {
|
|
|
|
t.Fatalf("bad: %#v", actual)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-08-19 00:41:12 +02:00
|
|
|
func TestResourceDataSetDependencies(t *testing.T) {
|
|
|
|
d := &ResourceData{}
|
2014-08-22 07:19:33 +02:00
|
|
|
d.SetId("foo")
|
2014-08-19 00:41:12 +02:00
|
|
|
d.SetDependencies([]terraform.ResourceDependency{
|
|
|
|
terraform.ResourceDependency{ID: "foo"},
|
|
|
|
})
|
|
|
|
|
|
|
|
expected := []terraform.ResourceDependency{
|
|
|
|
terraform.ResourceDependency{ID: "foo"},
|
|
|
|
}
|
|
|
|
|
|
|
|
actual := d.State()
|
|
|
|
if !reflect.DeepEqual(actual.Dependencies, expected) {
|
|
|
|
t.Fatalf("bad: %#v", actual)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-08-18 05:48:50 +02:00
|
|
|
func TestResourceDataSetId(t *testing.T) {
|
|
|
|
d := &ResourceData{}
|
|
|
|
d.SetId("foo")
|
|
|
|
|
|
|
|
actual := d.State()
|
|
|
|
if actual.ID != "foo" {
|
|
|
|
t.Fatalf("bad: %#v", actual)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestResourceDataSetId_clear(t *testing.T) {
|
|
|
|
d := &ResourceData{
|
|
|
|
state: &terraform.ResourceState{ID: "bar"},
|
|
|
|
}
|
|
|
|
d.SetId("")
|
|
|
|
|
|
|
|
actual := d.State()
|
2014-08-22 07:19:33 +02:00
|
|
|
if actual != nil {
|
2014-08-18 05:48:50 +02:00
|
|
|
t.Fatalf("bad: %#v", actual)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestResourceDataSetId_override(t *testing.T) {
|
|
|
|
d := &ResourceData{
|
|
|
|
state: &terraform.ResourceState{ID: "bar"},
|
|
|
|
}
|
|
|
|
d.SetId("foo")
|
|
|
|
|
|
|
|
actual := d.State()
|
|
|
|
if actual.ID != "foo" {
|
|
|
|
t.Fatalf("bad: %#v", actual)
|
|
|
|
}
|
|
|
|
}
|