mirror of https://github.com/pulumi/pulumi.git
270 lines
6.4 KiB
Go
270 lines
6.4 KiB
Go
// Copyright 2016-2021, Pulumi Corporation. All rights reserved.
|
|
|
|
package graph
|
|
|
|
import (
|
|
"testing"
|
|
|
|
"github.com/pulumi/pulumi/pkg/v3/resource/deploy/providers"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/resource"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/tokens"
|
|
"github.com/stretchr/testify/assert"
|
|
)
|
|
|
|
func NewProviderResource(pkg, name, id string, deps ...resource.URN) *resource.State {
|
|
t := providers.MakeProviderType(tokens.Package(pkg))
|
|
return &resource.State{
|
|
Type: t,
|
|
URN: resource.NewURN("test", "test", "", t, tokens.QName(name)),
|
|
ID: resource.ID(id),
|
|
Inputs: resource.PropertyMap{},
|
|
Outputs: resource.PropertyMap{},
|
|
Dependencies: deps,
|
|
}
|
|
}
|
|
|
|
func NewResource(name string, provider *resource.State, deps ...resource.URN) *resource.State {
|
|
prov := ""
|
|
if provider != nil {
|
|
p, err := providers.NewReference(provider.URN, provider.ID)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
prov = p.String()
|
|
}
|
|
|
|
t := tokens.Type("test:test:test")
|
|
return &resource.State{
|
|
Type: t,
|
|
URN: resource.NewURN("test", "test", "", t, tokens.QName(name)),
|
|
Inputs: resource.PropertyMap{},
|
|
Outputs: resource.PropertyMap{},
|
|
Dependencies: deps,
|
|
Provider: prov,
|
|
}
|
|
}
|
|
|
|
func TestBasicGraph(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
pA := NewProviderResource("test", "pA", "0")
|
|
a := NewResource("a", pA)
|
|
b := NewResource("b", pA, a.URN)
|
|
pB := NewProviderResource("test", "pB", "1", a.URN, b.URN)
|
|
c := NewResource("c", pB, a.URN)
|
|
d := NewResource("d", nil, b.URN)
|
|
|
|
dg := NewDependencyGraph([]*resource.State{
|
|
pA,
|
|
a,
|
|
b,
|
|
pB,
|
|
c,
|
|
d,
|
|
})
|
|
|
|
assert.Equal(t, []*resource.State{
|
|
a, b, pB, c, d,
|
|
}, dg.DependingOn(pA, nil, false))
|
|
|
|
assert.Equal(t, []*resource.State{
|
|
b, pB, c, d,
|
|
}, dg.DependingOn(a, nil, false))
|
|
|
|
assert.Equal(t, []*resource.State{
|
|
pB, c, d,
|
|
}, dg.DependingOn(b, nil, false))
|
|
|
|
assert.Equal(t, []*resource.State{
|
|
c,
|
|
}, dg.DependingOn(pB, nil, false))
|
|
|
|
assert.Nil(t, dg.DependingOn(c, nil, false))
|
|
assert.Nil(t, dg.DependingOn(d, nil, false))
|
|
|
|
assert.Nil(t, dg.DependingOn(pA, map[resource.URN]bool{
|
|
a.URN: true,
|
|
b.URN: true,
|
|
}, false))
|
|
|
|
assert.Equal(t, []*resource.State{
|
|
a, pB, c,
|
|
}, dg.DependingOn(pA, map[resource.URN]bool{
|
|
b.URN: true,
|
|
}, false))
|
|
|
|
assert.Equal(t, []*resource.State{
|
|
b, pB, c, d,
|
|
}, dg.DependingOn(pA, map[resource.URN]bool{
|
|
a.URN: true,
|
|
}, false))
|
|
|
|
assert.Equal(t, []*resource.State{
|
|
c,
|
|
}, dg.DependingOn(a, map[resource.URN]bool{
|
|
b.URN: true,
|
|
pB.URN: true,
|
|
}, false))
|
|
|
|
assert.Equal(t, []*resource.State{
|
|
pB, c,
|
|
}, dg.DependingOn(a, map[resource.URN]bool{
|
|
b.URN: true,
|
|
}, false))
|
|
|
|
assert.Equal(t, []*resource.State{
|
|
d,
|
|
}, dg.DependingOn(b, map[resource.URN]bool{
|
|
pB.URN: true,
|
|
}, false))
|
|
}
|
|
|
|
// Tests that we don't add the same node to the DependingOn set twice.
|
|
func TestGraphNoDuplicates(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
a := NewResource("a", nil)
|
|
b := NewResource("b", nil, a.URN)
|
|
c := NewResource("c", nil, a.URN)
|
|
d := NewResource("d", nil, b.URN, c.URN)
|
|
|
|
dg := NewDependencyGraph([]*resource.State{
|
|
a,
|
|
b,
|
|
c,
|
|
d,
|
|
})
|
|
|
|
assert.Equal(t, []*resource.State{
|
|
b, c, d,
|
|
}, dg.DependingOn(a, nil, false))
|
|
}
|
|
|
|
func TestDependenciesOf(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
pA := NewProviderResource("test", "pA", "0")
|
|
a := NewResource("a", pA)
|
|
b := NewResource("b", pA, a.URN)
|
|
c := NewResource("c", pA)
|
|
d := NewResource("d", pA)
|
|
d.Parent = a.URN
|
|
|
|
dg := NewDependencyGraph([]*resource.State{
|
|
pA,
|
|
a,
|
|
b,
|
|
c,
|
|
d,
|
|
})
|
|
|
|
aDepends := dg.DependenciesOf(a)
|
|
assert.True(t, aDepends[pA])
|
|
assert.False(t, aDepends[a])
|
|
assert.False(t, aDepends[b])
|
|
|
|
bDepends := dg.DependenciesOf(b)
|
|
assert.True(t, bDepends[pA])
|
|
assert.True(t, bDepends[a])
|
|
assert.False(t, bDepends[b])
|
|
|
|
cDepends := dg.DependenciesOf(c)
|
|
assert.True(t, cDepends[pA])
|
|
assert.False(t, cDepends[a])
|
|
assert.False(t, cDepends[b])
|
|
|
|
dDepends := dg.DependenciesOf(d)
|
|
assert.True(t, dDepends[pA])
|
|
assert.True(t, dDepends[a]) // due to A being the parent of D
|
|
assert.False(t, dDepends[b])
|
|
assert.False(t, dDepends[c])
|
|
}
|
|
|
|
func TestDependenciesOfRemoteComponents(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
aws := NewProviderResource("aws", "default", "0")
|
|
xyz := NewProviderResource("xyz", "default", "0")
|
|
first := NewResource("first", xyz)
|
|
firstNested := NewResource("firstNested", xyz)
|
|
firstNested.Parent = first.URN
|
|
sg := NewResource("sg", aws)
|
|
sg.Parent = firstNested.URN
|
|
second := NewResource("second", xyz)
|
|
rule := NewResource("rule", aws, first.URN)
|
|
rule.Parent = second.URN
|
|
|
|
dg := NewDependencyGraph([]*resource.State{
|
|
aws,
|
|
xyz,
|
|
first,
|
|
firstNested,
|
|
sg,
|
|
second,
|
|
rule,
|
|
})
|
|
|
|
ruleDepends := dg.DependenciesOf(rule)
|
|
assert.True(t, ruleDepends[first], "direct dependency")
|
|
assert.True(t, ruleDepends[firstNested], "child of dependency")
|
|
assert.True(t, ruleDepends[sg], "transitive child of dependency")
|
|
assert.True(t, ruleDepends[second], "parent")
|
|
assert.True(t, ruleDepends[aws], "provider")
|
|
assert.False(t, ruleDepends[xyz], "unrelated")
|
|
}
|
|
|
|
func TestDependenciesOfRemoteComponentsNoCycle(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
aws := NewProviderResource("aws", "default", "0")
|
|
parent := NewResource("parent", aws)
|
|
r := NewResource("r", aws, parent.URN)
|
|
child := NewResource("child", aws, r.URN)
|
|
child.Parent = parent.URN
|
|
|
|
dg := NewDependencyGraph([]*resource.State{
|
|
aws,
|
|
parent,
|
|
r,
|
|
child,
|
|
})
|
|
|
|
childDependencies := dg.DependenciesOf(child)
|
|
assert.True(t, childDependencies[aws])
|
|
assert.True(t, childDependencies[parent])
|
|
assert.True(t, childDependencies[r])
|
|
|
|
rDependencies := dg.DependenciesOf(r)
|
|
assert.True(t, rDependencies[aws])
|
|
assert.True(t, rDependencies[parent])
|
|
assert.False(t, rDependencies[child])
|
|
}
|
|
|
|
func TestTransitiveDependenciesOf(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
aws := NewProviderResource("aws", "default", "0")
|
|
parent := NewResource("parent", aws)
|
|
greatUncle := NewResource("greatUncle", aws)
|
|
uncle := NewResource("r", aws)
|
|
uncle.Parent = greatUncle.URN
|
|
child := NewResource("child", aws, uncle.URN)
|
|
child.Parent = parent.URN
|
|
baby := NewResource("baby", aws)
|
|
baby.Parent = child.URN
|
|
|
|
dg := NewDependencyGraph([]*resource.State{
|
|
aws,
|
|
parent,
|
|
greatUncle,
|
|
uncle,
|
|
child,
|
|
baby,
|
|
})
|
|
// <(relation)- as an alias for depends on via relation
|
|
// baby <(Parent)- child <(Dependency)- uncle <(Parent)- greatUncle <(Provider)- aws
|
|
set := dg.TransitiveDependenciesOf(baby)
|
|
assert.True(t, set[aws], "everything should depend on the provider")
|
|
assert.True(t, set[greatUncle], "child depends on greatUncle")
|
|
}
|