mirror of https://github.com/pulumi/pulumi.git
478 lines
13 KiB
Go
478 lines
13 KiB
Go
// Copyright 2016-2021, Pulumi Corporation.
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
package deploy
|
|
|
|
import (
|
|
"runtime"
|
|
"testing"
|
|
|
|
"github.com/pulumi/pulumi/pkg/v3/resource/deploy/deploytest"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/resource"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/resource/plugin"
|
|
"github.com/stretchr/testify/assert"
|
|
)
|
|
|
|
func TestIgnoreChanges(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
cases := []struct {
|
|
name string
|
|
oldInputs map[string]interface{}
|
|
newInputs map[string]interface{}
|
|
expected map[string]interface{}
|
|
ignoreChanges []string
|
|
expectFailure bool
|
|
}{
|
|
{
|
|
name: "Present in old and new sets",
|
|
oldInputs: map[string]interface{}{
|
|
"a": map[string]interface{}{
|
|
"b": "foo",
|
|
},
|
|
},
|
|
newInputs: map[string]interface{}{
|
|
"a": map[string]interface{}{
|
|
"b": "bar",
|
|
},
|
|
"c": 42,
|
|
},
|
|
expected: map[string]interface{}{
|
|
"a": map[string]interface{}{
|
|
"b": "foo",
|
|
},
|
|
"c": 42,
|
|
},
|
|
ignoreChanges: []string{"a.b"},
|
|
},
|
|
{
|
|
name: "Missing in new sets",
|
|
oldInputs: map[string]interface{}{
|
|
"a": map[string]interface{}{
|
|
"b": "foo",
|
|
},
|
|
},
|
|
newInputs: map[string]interface{}{
|
|
"a": map[string]interface{}{},
|
|
"c": 42,
|
|
},
|
|
expected: map[string]interface{}{
|
|
"a": map[string]interface{}{
|
|
"b": "foo",
|
|
},
|
|
"c": 42,
|
|
},
|
|
ignoreChanges: []string{"a.b"},
|
|
},
|
|
{
|
|
name: "Missing in old deletes",
|
|
oldInputs: map[string]interface{}{},
|
|
newInputs: map[string]interface{}{
|
|
"a": map[string]interface{}{
|
|
"b": "foo",
|
|
},
|
|
"c": 42,
|
|
},
|
|
expected: map[string]interface{}{
|
|
"a": map[string]interface{}{},
|
|
"c": 42,
|
|
},
|
|
ignoreChanges: []string{"a.b"},
|
|
},
|
|
{
|
|
name: "Missing keys in old and new are OK",
|
|
oldInputs: map[string]interface{}{},
|
|
newInputs: map[string]interface{}{},
|
|
ignoreChanges: []string{
|
|
"a",
|
|
"a.b",
|
|
"a.c[0]",
|
|
},
|
|
},
|
|
{
|
|
name: "Missing parent keys in only new fail",
|
|
oldInputs: map[string]interface{}{
|
|
"a": map[string]interface{}{
|
|
"b": "foo",
|
|
},
|
|
},
|
|
newInputs: map[string]interface{}{},
|
|
ignoreChanges: []string{"a.b"},
|
|
expectFailure: true,
|
|
},
|
|
}
|
|
|
|
for _, c := range cases {
|
|
c := c
|
|
t.Run(c.name, func(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
olds, news := resource.NewPropertyMapFromMap(c.oldInputs), resource.NewPropertyMapFromMap(c.newInputs)
|
|
|
|
expected := olds
|
|
if c.expected != nil {
|
|
expected = resource.NewPropertyMapFromMap(c.expected)
|
|
}
|
|
|
|
processed, err := processIgnoreChanges(news, olds, c.ignoreChanges)
|
|
if c.expectFailure {
|
|
assert.Error(t, err)
|
|
} else {
|
|
assert.NoError(t, err)
|
|
assert.Equal(t, expected, processed)
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestApplyReplaceOnChangesEmptyDetailedDiff(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
cases := []struct {
|
|
name string
|
|
diff plugin.DiffResult
|
|
replaceOnChanges []string
|
|
hasInitErrors bool
|
|
expected plugin.DiffResult
|
|
}{
|
|
{
|
|
name: "Empty diff and replaceOnChanges",
|
|
diff: plugin.DiffResult{},
|
|
replaceOnChanges: []string{},
|
|
hasInitErrors: false,
|
|
expected: plugin.DiffResult{},
|
|
},
|
|
{
|
|
name: "DiffSome and empty replaceOnChanges",
|
|
diff: plugin.DiffResult{
|
|
Changes: plugin.DiffSome,
|
|
ChangedKeys: []resource.PropertyKey{"a"},
|
|
},
|
|
replaceOnChanges: []string{},
|
|
hasInitErrors: false,
|
|
expected: plugin.DiffResult{
|
|
Changes: plugin.DiffSome,
|
|
ChangedKeys: []resource.PropertyKey{"a"},
|
|
},
|
|
},
|
|
{
|
|
name: "DiffSome and non-empty replaceOnChanges",
|
|
diff: plugin.DiffResult{
|
|
Changes: plugin.DiffSome,
|
|
ChangedKeys: []resource.PropertyKey{"a"},
|
|
},
|
|
replaceOnChanges: []string{"a"},
|
|
hasInitErrors: false,
|
|
expected: plugin.DiffResult{
|
|
Changes: plugin.DiffSome,
|
|
ChangedKeys: []resource.PropertyKey{"a"},
|
|
ReplaceKeys: []resource.PropertyKey{"a"},
|
|
},
|
|
},
|
|
{
|
|
name: "Empty diff and replaceOnChanges w/ init errors",
|
|
diff: plugin.DiffResult{},
|
|
replaceOnChanges: []string{},
|
|
hasInitErrors: true,
|
|
expected: plugin.DiffResult{},
|
|
},
|
|
{
|
|
name: "DiffSome and empty replaceOnChanges w/ init errors",
|
|
diff: plugin.DiffResult{
|
|
Changes: plugin.DiffSome,
|
|
ChangedKeys: []resource.PropertyKey{"a"},
|
|
},
|
|
replaceOnChanges: []string{},
|
|
hasInitErrors: true,
|
|
expected: plugin.DiffResult{
|
|
Changes: plugin.DiffSome,
|
|
ChangedKeys: []resource.PropertyKey{"a"},
|
|
},
|
|
},
|
|
{
|
|
name: "DiffSome and non-empty replaceOnChanges w/ init errors",
|
|
diff: plugin.DiffResult{
|
|
Changes: plugin.DiffSome,
|
|
ChangedKeys: []resource.PropertyKey{"a"},
|
|
},
|
|
replaceOnChanges: []string{"a"},
|
|
hasInitErrors: true,
|
|
expected: plugin.DiffResult{
|
|
Changes: plugin.DiffSome,
|
|
ChangedKeys: []resource.PropertyKey{"a"},
|
|
ReplaceKeys: []resource.PropertyKey{"a"},
|
|
},
|
|
},
|
|
{
|
|
name: "Empty diff and non-empty replaceOnChanges w/ init errors",
|
|
diff: plugin.DiffResult{},
|
|
replaceOnChanges: []string{"*"},
|
|
hasInitErrors: true,
|
|
expected: plugin.DiffResult{
|
|
Changes: plugin.DiffSome,
|
|
ReplaceKeys: []resource.PropertyKey{"#initerror"},
|
|
},
|
|
},
|
|
}
|
|
|
|
for _, c := range cases {
|
|
c := c
|
|
t.Run(c.name, func(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
newdiff, err := applyReplaceOnChanges(c.diff, c.replaceOnChanges, c.hasInitErrors)
|
|
assert.NoError(t, err)
|
|
assert.Equal(t, c.expected, newdiff)
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestEngineDiff(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
cases := []struct {
|
|
name string
|
|
oldInputs, newInputs resource.PropertyMap
|
|
ignoreChanges []string
|
|
expected []resource.PropertyKey
|
|
expectedChanges plugin.DiffChanges
|
|
}{
|
|
{
|
|
name: "Empty diff",
|
|
oldInputs: resource.NewPropertyMapFromMap(map[string]interface{}{
|
|
"val1": resource.NewPropertyValue(8),
|
|
"val2": resource.NewPropertyValue("hello"),
|
|
}),
|
|
newInputs: resource.NewPropertyMapFromMap(map[string]interface{}{
|
|
"val1": resource.NewPropertyValue(8),
|
|
"val2": resource.NewPropertyValue("hello"),
|
|
}),
|
|
expected: nil,
|
|
expectedChanges: plugin.DiffNone,
|
|
},
|
|
{
|
|
name: "All changes",
|
|
oldInputs: resource.NewPropertyMapFromMap(map[string]interface{}{
|
|
"val0": resource.NewPropertyValue(3.14),
|
|
}),
|
|
newInputs: resource.NewPropertyMapFromMap(map[string]interface{}{
|
|
"val1": resource.NewNumberProperty(42),
|
|
"val2": resource.NewPropertyValue("world"),
|
|
}),
|
|
expected: []resource.PropertyKey{"val0", "val1", "val2"},
|
|
expectedChanges: plugin.DiffSome,
|
|
},
|
|
{
|
|
name: "Some changes",
|
|
oldInputs: resource.NewPropertyMapFromMap(map[string]interface{}{
|
|
"val1": resource.NewPropertyValue(42),
|
|
}),
|
|
newInputs: resource.NewPropertyMapFromMap(map[string]interface{}{
|
|
"val1": resource.NewNumberProperty(42),
|
|
"val2": resource.NewPropertyValue("world"),
|
|
}),
|
|
expected: []resource.PropertyKey{"val2"},
|
|
expectedChanges: plugin.DiffSome,
|
|
},
|
|
{
|
|
name: "Ignore some changes",
|
|
oldInputs: resource.NewPropertyMapFromMap(map[string]interface{}{
|
|
"val1": resource.NewPropertyValue("hello"),
|
|
}),
|
|
newInputs: resource.NewPropertyMapFromMap(map[string]interface{}{
|
|
"val2": resource.NewPropertyValue(8),
|
|
}),
|
|
|
|
ignoreChanges: []string{"val1"},
|
|
expected: []resource.PropertyKey{"val2"},
|
|
expectedChanges: plugin.DiffSome,
|
|
},
|
|
{
|
|
name: "Ignore all changes",
|
|
oldInputs: resource.NewPropertyMapFromMap(map[string]interface{}{
|
|
"val1": resource.NewPropertyValue("hello"),
|
|
}),
|
|
newInputs: resource.NewPropertyMapFromMap(map[string]interface{}{
|
|
"val2": resource.NewPropertyValue(8),
|
|
}),
|
|
|
|
ignoreChanges: []string{"val1", "val2"},
|
|
expected: nil,
|
|
expectedChanges: plugin.DiffNone,
|
|
},
|
|
}
|
|
urn := resource.URN("urn:pulumi:dev::website-and-lambda::aws:s3/bucket:Bucket::my-bucket")
|
|
id := resource.ID("someid")
|
|
var oldOutputs resource.PropertyMap
|
|
allowUnknowns := false
|
|
provider := deploytest.Provider{}
|
|
for _, c := range cases {
|
|
c := c
|
|
t.Run(c.name, func(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
diff, err := diffResource(urn, id, c.oldInputs, oldOutputs, c.newInputs, &provider, allowUnknowns, c.ignoreChanges)
|
|
t.Logf("diff.ChangedKeys = %v", diff.ChangedKeys)
|
|
t.Logf("diff.StableKeys = %v", diff.StableKeys)
|
|
t.Logf("diff.ReplaceKeys = %v", diff.ReplaceKeys)
|
|
assert.NoError(t, err)
|
|
assert.Equal(t, c.expectedChanges, diff.Changes)
|
|
assert.EqualValues(t, c.expected, diff.ChangedKeys)
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestGenerateAliases(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
const (
|
|
project = "project"
|
|
stack = "stack"
|
|
)
|
|
|
|
parentTypeAlias := resource.CreateURN("myres", "test:resource:type2", "", project, stack)
|
|
parentNameAlias := resource.CreateURN("myres2", "test:resource:type", "", project, stack)
|
|
|
|
cases := []struct {
|
|
name string
|
|
parentAlias *resource.URN
|
|
childAliases []resource.Alias
|
|
expected []resource.URN
|
|
}{
|
|
{
|
|
name: "no aliases",
|
|
expected: nil,
|
|
},
|
|
{
|
|
name: "child alias (type), no parent aliases",
|
|
childAliases: []resource.Alias{
|
|
{Type: "test:resource:child2"},
|
|
},
|
|
expected: []resource.URN{
|
|
"urn:pulumi:stack::project::test:resource:type$test:resource:child2::myres-child",
|
|
},
|
|
},
|
|
{
|
|
name: "child alias (name), no parent aliases",
|
|
childAliases: []resource.Alias{
|
|
{Name: "child2"},
|
|
},
|
|
expected: []resource.URN{
|
|
"urn:pulumi:stack::project::test:resource:type$test:resource:child::child2",
|
|
},
|
|
},
|
|
{
|
|
name: "child alias (type, noParent), no parent aliases",
|
|
childAliases: []resource.Alias{
|
|
{
|
|
Type: "test:resource:child2",
|
|
NoParent: true,
|
|
},
|
|
},
|
|
expected: []resource.URN{
|
|
"urn:pulumi:stack::project::test:resource:child2::myres-child",
|
|
},
|
|
},
|
|
{
|
|
name: "child alias (type, parent), no parent aliases",
|
|
childAliases: []resource.Alias{
|
|
{
|
|
Type: "test:resource:child2",
|
|
Parent: resource.CreateURN("originalparent", "test:resource:original", "", project, stack),
|
|
},
|
|
},
|
|
expected: []resource.URN{
|
|
"urn:pulumi:stack::project::test:resource:original$test:resource:child2::myres-child",
|
|
},
|
|
},
|
|
{
|
|
name: "child alias (name), parent alias (type)",
|
|
parentAlias: &parentTypeAlias,
|
|
childAliases: []resource.Alias{
|
|
{Name: "myres-child2"},
|
|
},
|
|
expected: []resource.URN{
|
|
"urn:pulumi:stack::project::test:resource:type$test:resource:child::myres-child2",
|
|
"urn:pulumi:stack::project::test:resource:type2$test:resource:child::myres-child",
|
|
"urn:pulumi:stack::project::test:resource:type2$test:resource:child::myres-child2",
|
|
},
|
|
},
|
|
{
|
|
name: "child alias (name), parent alias (name)",
|
|
parentAlias: &parentNameAlias,
|
|
childAliases: []resource.Alias{
|
|
{Name: "myres-child2"},
|
|
},
|
|
expected: []resource.URN{
|
|
"urn:pulumi:stack::project::test:resource:type$test:resource:child::myres-child2",
|
|
"urn:pulumi:stack::project::test:resource:type$test:resource:child::myres2-child",
|
|
"urn:pulumi:stack::project::test:resource:type$test:resource:child::myres2-child2",
|
|
},
|
|
},
|
|
}
|
|
|
|
for _, tt := range cases {
|
|
tt := tt
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
parentURN := resource.CreateURN("myres", "test:resource:type", "", project, stack)
|
|
goal := &resource.Goal{
|
|
Parent: parentURN,
|
|
Name: "myres-child",
|
|
Type: "test:resource:child",
|
|
Aliases: tt.childAliases,
|
|
}
|
|
|
|
sg := newStepGenerator(&Deployment{
|
|
target: &Target{
|
|
Name: stack,
|
|
},
|
|
source: NewNullSource(project),
|
|
}, Options{}, NewUrnTargets(nil), NewUrnTargets(nil))
|
|
|
|
if tt.parentAlias != nil {
|
|
sg.aliases = map[resource.URN]resource.URN{
|
|
parentURN: *tt.parentAlias,
|
|
}
|
|
}
|
|
|
|
actual := sg.generateAliases(goal)
|
|
assert.Equal(t, tt.expected, actual)
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestDeleteProtectedErrorUsesCorrectQuotesOnOS(t *testing.T) {
|
|
t.Parallel()
|
|
err := deleteProtectedError{urn: "resource:urn"}
|
|
|
|
expectations := map[string]string{
|
|
`windows`: `"`,
|
|
`linux`: `'`,
|
|
`darwin`: `'`,
|
|
}
|
|
|
|
t.Run(runtime.GOOS, func(t *testing.T) {
|
|
t.Parallel()
|
|
gotErrMsg := err.Error()
|
|
contains, ok := expectations[runtime.GOOS]
|
|
if !ok {
|
|
t.Skipf("no quoting expectation for %s", runtime.GOOS)
|
|
return
|
|
}
|
|
assert.Contains(t, gotErrMsg, contains)
|
|
})
|
|
}
|