2016-09-13 12:56:37 -05:00
|
|
|
package terraform
|
|
|
|
|
|
|
|
import (
|
|
|
|
"reflect"
|
|
|
|
"strings"
|
|
|
|
"testing"
|
|
|
|
)
|
|
|
|
|
|
|
|
func TestApplyGraphBuilder_impl(t *testing.T) {
|
|
|
|
var _ GraphBuilder = new(ApplyGraphBuilder)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestApplyGraphBuilder(t *testing.T) {
|
|
|
|
diff := &Diff{
|
|
|
|
Modules: []*ModuleDiff{
|
|
|
|
&ModuleDiff{
|
|
|
|
Path: []string{"root"},
|
|
|
|
Resources: map[string]*InstanceDiff{
|
|
|
|
// Verify noop doesn't show up in graph
|
|
|
|
"aws_instance.noop": &InstanceDiff{},
|
|
|
|
|
|
|
|
"aws_instance.create": &InstanceDiff{
|
|
|
|
Attributes: map[string]*ResourceAttrDiff{
|
|
|
|
"name": &ResourceAttrDiff{
|
|
|
|
Old: "",
|
|
|
|
New: "foo",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
2016-09-15 20:30:11 -05:00
|
|
|
|
|
|
|
"aws_instance.other": &InstanceDiff{
|
|
|
|
Attributes: map[string]*ResourceAttrDiff{
|
|
|
|
"name": &ResourceAttrDiff{
|
|
|
|
Old: "",
|
|
|
|
New: "foo",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
2016-09-13 12:56:37 -05:00
|
|
|
},
|
|
|
|
},
|
2016-09-14 16:43:14 -05:00
|
|
|
|
|
|
|
&ModuleDiff{
|
|
|
|
Path: []string{"root", "child"},
|
|
|
|
Resources: map[string]*InstanceDiff{
|
|
|
|
"aws_instance.create": &InstanceDiff{
|
|
|
|
Attributes: map[string]*ResourceAttrDiff{
|
|
|
|
"name": &ResourceAttrDiff{
|
|
|
|
Old: "",
|
|
|
|
New: "foo",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
2016-09-15 20:31:13 -05:00
|
|
|
|
|
|
|
"aws_instance.other": &InstanceDiff{
|
|
|
|
Attributes: map[string]*ResourceAttrDiff{
|
|
|
|
"name": &ResourceAttrDiff{
|
|
|
|
Old: "",
|
|
|
|
New: "foo",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
2016-09-14 16:43:14 -05:00
|
|
|
},
|
|
|
|
},
|
2016-09-13 12:56:37 -05:00
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
b := &ApplyGraphBuilder{
|
2016-09-22 13:03:03 -05:00
|
|
|
Module: testModule(t, "graph-builder-apply-basic"),
|
|
|
|
Diff: diff,
|
|
|
|
Providers: []string{"aws"},
|
|
|
|
Provisioners: []string{"exec"},
|
|
|
|
DisableReduce: true,
|
2016-09-13 12:56:37 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
g, err := b.Build(RootModulePath)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if !reflect.DeepEqual(g.Path, RootModulePath) {
|
|
|
|
t.Fatalf("bad: %#v", g.Path)
|
|
|
|
}
|
|
|
|
|
|
|
|
actual := strings.TrimSpace(g.String())
|
|
|
|
expected := strings.TrimSpace(testApplyGraphBuilderStr)
|
|
|
|
if actual != expected {
|
|
|
|
t.Fatalf("bad: %s", actual)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
terraform: apply resource must depend on destroy deps
Fixes #10440
This updates the behavior of "apply" resources to depend on the
destroy versions of their dependencies.
We make an exception to this behavior when the "apply" resource is CBD.
This is odd and not 100% correct, but it mimics the behavior of the
legacy graphs and avoids us having to do major core work to support the
100% correct solution.
I'll explain this in examples...
Given the following configuration:
resource "null_resource" "a" {
count = "${var.count}"
}
resource "null_resource" "b" {
triggers { key = "${join(",", null_resource.a.*.id)}" }
}
Assume we've successfully created this configuration with count = 2.
When going from count = 2 to count = 1, `null_resource.b` should wait
for `null_resource.a.1` to destroy.
If it doesn't, then it is a race: depending when we interpolate the
`triggers.key` attribute of `null_resource.b`, we may get 1 value or 2.
If `null_resource.a.1` is destroyed, we'll get 1. Otherwise, we'll get
2. This was the root cause of #10440
In the legacy graphs, `null_resource.b` would depend on the destruction
of any `null_resource.a` (orphans, tainted, anything!). This would
ensure proper ordering. We mimic that behavior here.
The difference is CBD. If `null_resource.b` has CBD enabled, then the
ordering **in the legacy graph** becomes:
1. null_resource.b (create)
2. null_resource.b (destroy)
3. null_resource.a (destroy)
In this case, the update would always have 2 values for `triggers.key`,
even though we were destroying a resource later! This scenario required
two `terraform apply` operations.
This is what the CBD check is for in this PR. We do this to mimic the
behavior of the legacy graph.
The correct solution to do one day is to allow splat references
(`null_resource.a.*.id`) to happen in parallel and only read up to to
the `count` amount in the state. This requires some fairly significant
work close to the 0.8 release date, so we can defer this to later and
adopt the 0.7.x behavior for now.
2016-12-04 01:44:09 -06:00
|
|
|
// This tests the ordering of two resources that are both CBD that
|
|
|
|
// require destroy/create.
|
|
|
|
func TestApplyGraphBuilder_doubleCBD(t *testing.T) {
|
|
|
|
diff := &Diff{
|
|
|
|
Modules: []*ModuleDiff{
|
|
|
|
&ModuleDiff{
|
|
|
|
Path: []string{"root"},
|
|
|
|
Resources: map[string]*InstanceDiff{
|
|
|
|
"aws_instance.A": &InstanceDiff{
|
|
|
|
Destroy: true,
|
|
|
|
Attributes: map[string]*ResourceAttrDiff{
|
|
|
|
"name": &ResourceAttrDiff{
|
|
|
|
Old: "",
|
|
|
|
New: "foo",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
"aws_instance.B": &InstanceDiff{
|
|
|
|
Destroy: true,
|
|
|
|
Attributes: map[string]*ResourceAttrDiff{
|
|
|
|
"name": &ResourceAttrDiff{
|
|
|
|
Old: "",
|
|
|
|
New: "foo",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
b := &ApplyGraphBuilder{
|
|
|
|
Module: testModule(t, "graph-builder-apply-double-cbd"),
|
|
|
|
Diff: diff,
|
|
|
|
Providers: []string{"aws"},
|
|
|
|
Provisioners: []string{"exec"},
|
|
|
|
DisableReduce: true,
|
|
|
|
}
|
|
|
|
|
|
|
|
g, err := b.Build(RootModulePath)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if !reflect.DeepEqual(g.Path, RootModulePath) {
|
|
|
|
t.Fatalf("bad: %#v", g.Path)
|
|
|
|
}
|
|
|
|
|
|
|
|
actual := strings.TrimSpace(g.String())
|
|
|
|
expected := strings.TrimSpace(testApplyGraphBuilderDoubleCBDStr)
|
|
|
|
if actual != expected {
|
|
|
|
t.Fatalf("bad: %s", actual)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// This tests the ordering of destroying a single count of a resource.
|
|
|
|
func TestApplyGraphBuilder_destroyCount(t *testing.T) {
|
|
|
|
diff := &Diff{
|
|
|
|
Modules: []*ModuleDiff{
|
|
|
|
&ModuleDiff{
|
|
|
|
Path: []string{"root"},
|
|
|
|
Resources: map[string]*InstanceDiff{
|
|
|
|
"aws_instance.A.1": &InstanceDiff{
|
|
|
|
Destroy: true,
|
|
|
|
},
|
|
|
|
|
|
|
|
"aws_instance.B": &InstanceDiff{
|
|
|
|
Attributes: map[string]*ResourceAttrDiff{
|
|
|
|
"name": &ResourceAttrDiff{
|
|
|
|
Old: "",
|
|
|
|
New: "foo",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
b := &ApplyGraphBuilder{
|
|
|
|
Module: testModule(t, "graph-builder-apply-count"),
|
|
|
|
Diff: diff,
|
|
|
|
Providers: []string{"aws"},
|
|
|
|
Provisioners: []string{"exec"},
|
|
|
|
DisableReduce: true,
|
|
|
|
}
|
|
|
|
|
|
|
|
g, err := b.Build(RootModulePath)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if !reflect.DeepEqual(g.Path, RootModulePath) {
|
|
|
|
t.Fatalf("bad: %#v", g.Path)
|
|
|
|
}
|
|
|
|
|
|
|
|
actual := strings.TrimSpace(g.String())
|
|
|
|
expected := strings.TrimSpace(testApplyGraphBuilderDestroyCountStr)
|
|
|
|
if actual != expected {
|
|
|
|
t.Fatalf("bad: %s", actual)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-12-14 23:39:13 -06:00
|
|
|
func TestApplyGraphBuilder_moduleDestroy(t *testing.T) {
|
|
|
|
diff := &Diff{
|
|
|
|
Modules: []*ModuleDiff{
|
|
|
|
&ModuleDiff{
|
|
|
|
Path: []string{"root", "A"},
|
|
|
|
Resources: map[string]*InstanceDiff{
|
|
|
|
"null_resource.foo": &InstanceDiff{
|
|
|
|
Destroy: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
&ModuleDiff{
|
|
|
|
Path: []string{"root", "B"},
|
|
|
|
Resources: map[string]*InstanceDiff{
|
|
|
|
"null_resource.foo": &InstanceDiff{
|
|
|
|
Destroy: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
b := &ApplyGraphBuilder{
|
|
|
|
Module: testModule(t, "graph-builder-apply-module-destroy"),
|
|
|
|
Diff: diff,
|
|
|
|
Providers: []string{"null"},
|
|
|
|
}
|
|
|
|
|
|
|
|
g, err := b.Build(RootModulePath)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
testGraphHappensBefore(
|
|
|
|
t, g,
|
|
|
|
"module.B.null_resource.foo (destroy)",
|
|
|
|
"module.A.null_resource.foo (destroy)")
|
|
|
|
}
|
|
|
|
|
2016-09-13 12:56:37 -05:00
|
|
|
const testApplyGraphBuilderStr = `
|
|
|
|
aws_instance.create
|
2016-09-13 19:11:34 -05:00
|
|
|
provider.aws
|
2016-09-15 20:30:11 -05:00
|
|
|
aws_instance.other
|
|
|
|
aws_instance.create
|
|
|
|
provider.aws
|
2016-09-22 13:03:03 -05:00
|
|
|
meta.count-boundary (count boundary fixup)
|
|
|
|
aws_instance.create
|
|
|
|
aws_instance.other
|
|
|
|
module.child.aws_instance.create
|
|
|
|
module.child.aws_instance.other
|
|
|
|
module.child.provider.aws
|
2016-11-03 12:25:11 -05:00
|
|
|
module.child.provisioner.exec
|
2016-09-22 13:03:03 -05:00
|
|
|
provider.aws
|
2016-09-14 16:43:14 -05:00
|
|
|
module.child.aws_instance.create
|
|
|
|
module.child.provider.aws
|
2016-11-03 12:25:11 -05:00
|
|
|
module.child.provisioner.exec
|
2016-09-15 20:31:13 -05:00
|
|
|
module.child.aws_instance.other
|
|
|
|
module.child.aws_instance.create
|
|
|
|
module.child.provider.aws
|
2016-09-14 16:43:14 -05:00
|
|
|
module.child.provider.aws
|
|
|
|
provider.aws
|
2016-11-03 12:25:11 -05:00
|
|
|
module.child.provisioner.exec
|
2016-09-13 19:11:34 -05:00
|
|
|
provider.aws
|
2016-09-13 12:56:37 -05:00
|
|
|
`
|
terraform: apply resource must depend on destroy deps
Fixes #10440
This updates the behavior of "apply" resources to depend on the
destroy versions of their dependencies.
We make an exception to this behavior when the "apply" resource is CBD.
This is odd and not 100% correct, but it mimics the behavior of the
legacy graphs and avoids us having to do major core work to support the
100% correct solution.
I'll explain this in examples...
Given the following configuration:
resource "null_resource" "a" {
count = "${var.count}"
}
resource "null_resource" "b" {
triggers { key = "${join(",", null_resource.a.*.id)}" }
}
Assume we've successfully created this configuration with count = 2.
When going from count = 2 to count = 1, `null_resource.b` should wait
for `null_resource.a.1` to destroy.
If it doesn't, then it is a race: depending when we interpolate the
`triggers.key` attribute of `null_resource.b`, we may get 1 value or 2.
If `null_resource.a.1` is destroyed, we'll get 1. Otherwise, we'll get
2. This was the root cause of #10440
In the legacy graphs, `null_resource.b` would depend on the destruction
of any `null_resource.a` (orphans, tainted, anything!). This would
ensure proper ordering. We mimic that behavior here.
The difference is CBD. If `null_resource.b` has CBD enabled, then the
ordering **in the legacy graph** becomes:
1. null_resource.b (create)
2. null_resource.b (destroy)
3. null_resource.a (destroy)
In this case, the update would always have 2 values for `triggers.key`,
even though we were destroying a resource later! This scenario required
two `terraform apply` operations.
This is what the CBD check is for in this PR. We do this to mimic the
behavior of the legacy graph.
The correct solution to do one day is to allow splat references
(`null_resource.a.*.id`) to happen in parallel and only read up to to
the `count` amount in the state. This requires some fairly significant
work close to the 0.8 release date, so we can defer this to later and
adopt the 0.7.x behavior for now.
2016-12-04 01:44:09 -06:00
|
|
|
|
|
|
|
const testApplyGraphBuilderDoubleCBDStr = `
|
|
|
|
aws_instance.A
|
|
|
|
provider.aws
|
|
|
|
aws_instance.A (destroy)
|
|
|
|
aws_instance.A
|
|
|
|
aws_instance.B
|
|
|
|
aws_instance.B (destroy)
|
|
|
|
provider.aws
|
|
|
|
aws_instance.B
|
|
|
|
aws_instance.A
|
|
|
|
provider.aws
|
|
|
|
aws_instance.B (destroy)
|
|
|
|
aws_instance.B
|
|
|
|
provider.aws
|
|
|
|
meta.count-boundary (count boundary fixup)
|
|
|
|
aws_instance.A
|
|
|
|
aws_instance.A (destroy)
|
|
|
|
aws_instance.B
|
|
|
|
aws_instance.B (destroy)
|
|
|
|
provider.aws
|
|
|
|
provider.aws
|
|
|
|
`
|
|
|
|
|
|
|
|
const testApplyGraphBuilderDestroyCountStr = `
|
|
|
|
aws_instance.A[1] (destroy)
|
|
|
|
provider.aws
|
|
|
|
aws_instance.B
|
|
|
|
aws_instance.A[1] (destroy)
|
|
|
|
provider.aws
|
|
|
|
meta.count-boundary (count boundary fixup)
|
|
|
|
aws_instance.A[1] (destroy)
|
|
|
|
aws_instance.B
|
|
|
|
provider.aws
|
|
|
|
provider.aws
|
|
|
|
`
|