2016-09-15 20:41:09 -05:00
|
|
|
package terraform
|
|
|
|
|
|
|
|
import (
|
2016-11-04 20:40:09 -05:00
|
|
|
"reflect"
|
|
|
|
"sort"
|
2016-09-15 20:41:09 -05:00
|
|
|
"strings"
|
|
|
|
"testing"
|
2016-11-04 20:40:09 -05:00
|
|
|
|
2021-05-17 14:00:50 -05:00
|
|
|
"github.com/hashicorp/terraform/internal/addrs"
|
2021-05-17 11:30:37 -05:00
|
|
|
"github.com/hashicorp/terraform/internal/dag"
|
2016-09-15 20:41:09 -05:00
|
|
|
)
|
|
|
|
|
|
|
|
func TestReferenceTransformer_simple(t *testing.T) {
|
2018-05-04 21:24:06 -05:00
|
|
|
g := Graph{Path: addrs.RootModuleInstance}
|
2016-09-15 20:41:09 -05:00
|
|
|
g.Add(&graphNodeRefParentTest{
|
|
|
|
NameValue: "A",
|
|
|
|
Names: []string{"A"},
|
|
|
|
})
|
|
|
|
g.Add(&graphNodeRefChildTest{
|
|
|
|
NameValue: "B",
|
|
|
|
Refs: []string{"A"},
|
|
|
|
})
|
|
|
|
|
|
|
|
tf := &ReferenceTransformer{}
|
|
|
|
if err := tf.Transform(&g); err != nil {
|
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
actual := strings.TrimSpace(g.String())
|
|
|
|
expected := strings.TrimSpace(testTransformRefBasicStr)
|
|
|
|
if actual != expected {
|
2018-05-07 19:58:43 -05:00
|
|
|
t.Fatalf("wrong result\n\ngot:\n%s\n\nwant:\n%s", actual, expected)
|
2016-09-15 20:41:09 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-09-18 23:31:03 -05:00
|
|
|
func TestReferenceTransformer_self(t *testing.T) {
|
2018-05-04 21:24:06 -05:00
|
|
|
g := Graph{Path: addrs.RootModuleInstance}
|
2016-09-18 23:31:03 -05:00
|
|
|
g.Add(&graphNodeRefParentTest{
|
|
|
|
NameValue: "A",
|
|
|
|
Names: []string{"A"},
|
|
|
|
})
|
|
|
|
g.Add(&graphNodeRefChildTest{
|
|
|
|
NameValue: "B",
|
|
|
|
Refs: []string{"A", "B"},
|
|
|
|
})
|
|
|
|
|
|
|
|
tf := &ReferenceTransformer{}
|
|
|
|
if err := tf.Transform(&g); err != nil {
|
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
actual := strings.TrimSpace(g.String())
|
|
|
|
expected := strings.TrimSpace(testTransformRefBasicStr)
|
|
|
|
if actual != expected {
|
2018-05-07 19:58:43 -05:00
|
|
|
t.Fatalf("wrong result\n\ngot:\n%s\n\nwant:\n%s", actual, expected)
|
2016-09-18 23:31:03 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-09-15 20:41:09 -05:00
|
|
|
func TestReferenceTransformer_path(t *testing.T) {
|
2018-05-04 21:24:06 -05:00
|
|
|
g := Graph{Path: addrs.RootModuleInstance}
|
2016-09-15 20:41:09 -05:00
|
|
|
g.Add(&graphNodeRefParentTest{
|
|
|
|
NameValue: "A",
|
|
|
|
Names: []string{"A"},
|
|
|
|
})
|
|
|
|
g.Add(&graphNodeRefChildTest{
|
|
|
|
NameValue: "B",
|
|
|
|
Refs: []string{"A"},
|
|
|
|
})
|
|
|
|
g.Add(&graphNodeRefParentTest{
|
|
|
|
NameValue: "child.A",
|
2020-11-13 12:59:11 -06:00
|
|
|
PathValue: addrs.ModuleInstance{addrs.ModuleInstanceStep{Name: "child"}},
|
2016-09-15 20:41:09 -05:00
|
|
|
Names: []string{"A"},
|
|
|
|
})
|
|
|
|
g.Add(&graphNodeRefChildTest{
|
|
|
|
NameValue: "child.B",
|
2020-11-13 12:59:11 -06:00
|
|
|
PathValue: addrs.ModuleInstance{addrs.ModuleInstanceStep{Name: "child"}},
|
2016-09-15 20:41:09 -05:00
|
|
|
Refs: []string{"A"},
|
|
|
|
})
|
|
|
|
|
|
|
|
tf := &ReferenceTransformer{}
|
|
|
|
if err := tf.Transform(&g); err != nil {
|
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
actual := strings.TrimSpace(g.String())
|
|
|
|
expected := strings.TrimSpace(testTransformRefPathStr)
|
|
|
|
if actual != expected {
|
2018-05-07 19:58:43 -05:00
|
|
|
t.Fatalf("wrong result\n\ngot:\n%s\n\nwant:\n%s", actual, expected)
|
2016-11-12 10:24:09 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-07-16 18:11:08 -05:00
|
|
|
func TestReferenceTransformer_resourceInstances(t *testing.T) {
|
|
|
|
// Our reference analyses are all done based on unexpanded addresses
|
|
|
|
// so that we can use this transformer both in the plan graph (where things
|
|
|
|
// are not expanded yet) and the apply graph (where resource instances are
|
|
|
|
// pre-expanded but nothing else is.)
|
|
|
|
// However, that would make the result too conservative about instances
|
|
|
|
// of the same resource in different instances of the same module, so we
|
|
|
|
// make an exception for that situation in particular, keeping references
|
|
|
|
// between resource instances segregated by their containing module
|
|
|
|
// instance.
|
|
|
|
g := Graph{Path: addrs.RootModuleInstance}
|
|
|
|
moduleInsts := []addrs.ModuleInstance{
|
|
|
|
{
|
|
|
|
{
|
|
|
|
Name: "foo", InstanceKey: addrs.IntKey(0),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
{
|
|
|
|
Name: "foo", InstanceKey: addrs.IntKey(1),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
resourceAs := make([]addrs.AbsResourceInstance, len(moduleInsts))
|
|
|
|
for i, moduleInst := range moduleInsts {
|
|
|
|
resourceAs[i] = addrs.Resource{
|
|
|
|
Mode: addrs.ManagedResourceMode,
|
|
|
|
Type: "thing",
|
|
|
|
Name: "a",
|
|
|
|
}.Instance(addrs.NoKey).Absolute(moduleInst)
|
|
|
|
}
|
|
|
|
resourceBs := make([]addrs.AbsResourceInstance, len(moduleInsts))
|
|
|
|
for i, moduleInst := range moduleInsts {
|
|
|
|
resourceBs[i] = addrs.Resource{
|
|
|
|
Mode: addrs.ManagedResourceMode,
|
|
|
|
Type: "thing",
|
|
|
|
Name: "b",
|
|
|
|
}.Instance(addrs.NoKey).Absolute(moduleInst)
|
|
|
|
}
|
|
|
|
g.Add(&graphNodeFakeResourceInstance{
|
|
|
|
Addr: resourceAs[0],
|
|
|
|
})
|
|
|
|
g.Add(&graphNodeFakeResourceInstance{
|
|
|
|
Addr: resourceBs[0],
|
|
|
|
Refs: []*addrs.Reference{
|
|
|
|
{
|
|
|
|
Subject: resourceAs[0].Resource,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
})
|
|
|
|
g.Add(&graphNodeFakeResourceInstance{
|
|
|
|
Addr: resourceAs[1],
|
|
|
|
})
|
|
|
|
g.Add(&graphNodeFakeResourceInstance{
|
|
|
|
Addr: resourceBs[1],
|
|
|
|
Refs: []*addrs.Reference{
|
|
|
|
{
|
|
|
|
Subject: resourceAs[1].Resource,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
})
|
|
|
|
|
|
|
|
tf := &ReferenceTransformer{}
|
|
|
|
if err := tf.Transform(&g); err != nil {
|
|
|
|
t.Fatalf("unexpected error: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Resource B should be connected to resource A in each module instance,
|
|
|
|
// but there should be no connections between the two module instances.
|
|
|
|
actual := strings.TrimSpace(g.String())
|
|
|
|
expected := strings.TrimSpace(`
|
|
|
|
module.foo[0].thing.a
|
|
|
|
module.foo[0].thing.b
|
|
|
|
module.foo[0].thing.a
|
|
|
|
module.foo[1].thing.a
|
|
|
|
module.foo[1].thing.b
|
|
|
|
module.foo[1].thing.a
|
|
|
|
`)
|
|
|
|
if actual != expected {
|
|
|
|
t.Fatalf("wrong result\n\ngot:\n%s\n\nwant:\n%s", actual, expected)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-11-04 20:40:09 -05:00
|
|
|
func TestReferenceMapReferences(t *testing.T) {
|
|
|
|
cases := map[string]struct {
|
|
|
|
Nodes []dag.Vertex
|
|
|
|
Check dag.Vertex
|
|
|
|
Result []string
|
|
|
|
}{
|
|
|
|
"simple": {
|
|
|
|
Nodes: []dag.Vertex{
|
|
|
|
&graphNodeRefParentTest{
|
|
|
|
NameValue: "A",
|
|
|
|
Names: []string{"A"},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
Check: &graphNodeRefChildTest{
|
|
|
|
NameValue: "foo",
|
|
|
|
Refs: []string{"A"},
|
|
|
|
},
|
|
|
|
Result: []string{"A"},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for tn, tc := range cases {
|
|
|
|
t.Run(tn, func(t *testing.T) {
|
|
|
|
rm := NewReferenceMap(tc.Nodes)
|
2020-02-24 16:42:32 -06:00
|
|
|
result := rm.References(tc.Check)
|
2016-11-04 20:58:03 -05:00
|
|
|
|
2016-11-04 20:40:09 -05:00
|
|
|
var resultStr []string
|
|
|
|
for _, v := range result {
|
|
|
|
resultStr = append(resultStr, dag.VertexName(v))
|
|
|
|
}
|
|
|
|
|
|
|
|
sort.Strings(resultStr)
|
|
|
|
sort.Strings(tc.Result)
|
|
|
|
if !reflect.DeepEqual(resultStr, tc.Result) {
|
|
|
|
t.Fatalf("bad: %#v", resultStr)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-09-15 20:41:09 -05:00
|
|
|
type graphNodeRefParentTest struct {
|
|
|
|
NameValue string
|
2020-11-13 12:59:11 -06:00
|
|
|
PathValue addrs.ModuleInstance
|
2016-09-15 20:41:09 -05:00
|
|
|
Names []string
|
|
|
|
}
|
|
|
|
|
2018-05-07 19:58:43 -05:00
|
|
|
var _ GraphNodeReferenceable = (*graphNodeRefParentTest)(nil)
|
|
|
|
|
|
|
|
func (n *graphNodeRefParentTest) Name() string {
|
|
|
|
return n.NameValue
|
|
|
|
}
|
|
|
|
|
|
|
|
func (n *graphNodeRefParentTest) ReferenceableAddrs() []addrs.Referenceable {
|
|
|
|
ret := make([]addrs.Referenceable, len(n.Names))
|
|
|
|
for i, name := range n.Names {
|
|
|
|
ret[i] = addrs.LocalValue{Name: name}
|
|
|
|
}
|
|
|
|
return ret
|
|
|
|
}
|
|
|
|
|
|
|
|
func (n *graphNodeRefParentTest) Path() addrs.ModuleInstance {
|
2020-11-13 12:59:11 -06:00
|
|
|
return n.PathValue
|
2018-05-07 19:58:43 -05:00
|
|
|
}
|
2016-09-15 20:41:09 -05:00
|
|
|
|
2020-03-04 20:00:16 -06:00
|
|
|
func (n *graphNodeRefParentTest) ModulePath() addrs.Module {
|
2020-11-13 12:59:11 -06:00
|
|
|
return n.PathValue.Module()
|
2020-03-04 20:00:16 -06:00
|
|
|
}
|
|
|
|
|
2016-09-15 20:41:09 -05:00
|
|
|
type graphNodeRefChildTest struct {
|
|
|
|
NameValue string
|
2020-11-13 12:59:11 -06:00
|
|
|
PathValue addrs.ModuleInstance
|
2016-09-15 20:41:09 -05:00
|
|
|
Refs []string
|
|
|
|
}
|
|
|
|
|
2018-05-07 19:58:43 -05:00
|
|
|
var _ GraphNodeReferencer = (*graphNodeRefChildTest)(nil)
|
|
|
|
|
|
|
|
func (n *graphNodeRefChildTest) Name() string {
|
|
|
|
return n.NameValue
|
|
|
|
}
|
|
|
|
|
|
|
|
func (n *graphNodeRefChildTest) References() []*addrs.Reference {
|
|
|
|
ret := make([]*addrs.Reference, len(n.Refs))
|
|
|
|
for i, name := range n.Refs {
|
|
|
|
ret[i] = &addrs.Reference{
|
|
|
|
Subject: addrs.LocalValue{Name: name},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return ret
|
|
|
|
}
|
|
|
|
|
|
|
|
func (n *graphNodeRefChildTest) Path() addrs.ModuleInstance {
|
2020-11-13 12:59:11 -06:00
|
|
|
return n.PathValue
|
2018-05-07 19:58:43 -05:00
|
|
|
}
|
2016-09-15 20:41:09 -05:00
|
|
|
|
2020-03-04 20:00:16 -06:00
|
|
|
func (n *graphNodeRefChildTest) ModulePath() addrs.Module {
|
2020-11-13 12:59:11 -06:00
|
|
|
return n.PathValue.Module()
|
2020-03-04 20:00:16 -06:00
|
|
|
}
|
|
|
|
|
2020-07-16 18:11:08 -05:00
|
|
|
type graphNodeFakeResourceInstance struct {
|
|
|
|
Addr addrs.AbsResourceInstance
|
|
|
|
Refs []*addrs.Reference
|
|
|
|
}
|
|
|
|
|
|
|
|
var _ GraphNodeResourceInstance = (*graphNodeFakeResourceInstance)(nil)
|
|
|
|
var _ GraphNodeReferenceable = (*graphNodeFakeResourceInstance)(nil)
|
|
|
|
var _ GraphNodeReferencer = (*graphNodeFakeResourceInstance)(nil)
|
|
|
|
|
|
|
|
func (n *graphNodeFakeResourceInstance) ResourceInstanceAddr() addrs.AbsResourceInstance {
|
|
|
|
return n.Addr
|
|
|
|
}
|
|
|
|
|
|
|
|
func (n *graphNodeFakeResourceInstance) ModulePath() addrs.Module {
|
|
|
|
return n.Addr.Module.Module()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (n *graphNodeFakeResourceInstance) ReferenceableAddrs() []addrs.Referenceable {
|
|
|
|
return []addrs.Referenceable{n.Addr.Resource}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (n *graphNodeFakeResourceInstance) References() []*addrs.Reference {
|
|
|
|
return n.Refs
|
|
|
|
}
|
|
|
|
|
|
|
|
func (n *graphNodeFakeResourceInstance) StateDependencies() []addrs.ConfigResource {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (n *graphNodeFakeResourceInstance) String() string {
|
|
|
|
return n.Addr.String()
|
|
|
|
}
|
|
|
|
|
2016-09-15 20:41:09 -05:00
|
|
|
const testTransformRefBasicStr = `
|
|
|
|
A
|
|
|
|
B
|
|
|
|
A
|
|
|
|
`
|
|
|
|
|
|
|
|
const testTransformRefPathStr = `
|
|
|
|
A
|
|
|
|
B
|
|
|
|
A
|
|
|
|
child.A
|
|
|
|
child.B
|
|
|
|
child.A
|
|
|
|
`
|