mirror of https://github.com/pulumi/pulumi.git
348 lines
34 KiB
Go
348 lines
34 KiB
Go
// Copyright 2016-2018, Pulumi Corporation. All rights reserved.
|
|
|
|
package ints
|
|
|
|
import (
|
|
"bytes"
|
|
"regexp"
|
|
"strings"
|
|
"testing"
|
|
"text/template"
|
|
"unicode"
|
|
|
|
"github.com/pmezard/go-difflib/difflib"
|
|
"github.com/pulumi/pulumi/pkg/resource"
|
|
"github.com/pulumi/pulumi/pkg/testing/integration"
|
|
"github.com/pulumi/pulumi/pkg/tokens"
|
|
"github.com/stretchr/testify/assert"
|
|
)
|
|
|
|
// TestDiffs tests many combinations of creates, updates, deletes, replacements, and checks the
|
|
// output of the command against an expected baseline.
|
|
func TestDiffs(t *testing.T) {
|
|
var buf bytes.Buffer
|
|
|
|
opts := integration.ProgramTestOptions{
|
|
Dir: "step1",
|
|
Dependencies: []string{"@pulumi/pulumi"},
|
|
Quick: true,
|
|
UpdateCommandlineFlags: []string{"--color=raw", "--non-interactive", "--diff"},
|
|
ExtraRuntimeValidation: func(t *testing.T, stack integration.RuntimeValidationStackInfo) {
|
|
assert.NotNil(t, stack.Deployment)
|
|
assert.Equal(t, 5, len(stack.Deployment.Resources))
|
|
stackRes := stack.Deployment.Resources[0]
|
|
assert.Equal(t, resource.RootStackType, stackRes.URN.Type())
|
|
a := stack.Deployment.Resources[1]
|
|
assert.Equal(t, "a", string(a.URN.Name()))
|
|
b := stack.Deployment.Resources[2]
|
|
assert.Equal(t, "b", string(b.URN.Name()))
|
|
c := stack.Deployment.Resources[3]
|
|
assert.Equal(t, "c", string(c.URN.Name()))
|
|
d := stack.Deployment.Resources[4]
|
|
assert.Equal(t, "d", string(d.URN.Name()))
|
|
},
|
|
EditDirs: []integration.EditDir{
|
|
{
|
|
Dir: "step2",
|
|
Additive: true,
|
|
Stdout: &buf,
|
|
Verbose: true,
|
|
ExtraRuntimeValidation: func(t *testing.T, stack integration.RuntimeValidationStackInfo) {
|
|
assert.NotNil(t, stack.Deployment)
|
|
assert.Equal(t, 5, len(stack.Deployment.Resources))
|
|
stackRes := stack.Deployment.Resources[0]
|
|
assert.Equal(t, resource.RootStackType, stackRes.URN.Type())
|
|
a := stack.Deployment.Resources[1]
|
|
assert.Equal(t, "a", string(a.URN.Name()))
|
|
b := stack.Deployment.Resources[2]
|
|
assert.Equal(t, "b", string(b.URN.Name()))
|
|
c := stack.Deployment.Resources[3]
|
|
assert.Equal(t, "c", string(c.URN.Name()))
|
|
e := stack.Deployment.Resources[4]
|
|
assert.Equal(t, "e", string(e.URN.Name()))
|
|
|
|
expected :=
|
|
fillStackName(`<unchanged>Performing changes:
|
|
* pulumi:pulumi:Stack: (same)
|
|
[urn=urn:pulumi:{{.StackName}}::steps::pulumi:pulumi:Stack::steps-{{.StackName}}]</unchanged>
|
|
<added>+ pulumi-nodejs:dynamic:Resource: (create)
|
|
[urn=urn:pulumi:{{.StackName}}::steps::pulumi-nodejs:dynamic:Resource::e]
|
|
__provider: "exports.handler = __f0;\n\nvar __provider_proto = {};\n__f1.prototype = __provider_proto;\n__f1.instance = __provider;\nObject.defineProperty(__provider_proto, \"constructor\", { configurable: true, writable: true, value: __f1 });\nObject.defineProperty(__provider_proto, \"diff\", { configurable: true, writable: true, value: __f2 });\nObject.defineProperty(__provider_proto, \"create\", { configurable: true, writable: true, value: __f4 });\nObject.defineProperty(__provider_proto, \"update\", { configurable: true, writable: true, value: __f5 });\nObject.defineProperty(__provider_proto, \"delete\", { configurable: true, writable: true, value: __f6 });\nObject.defineProperty(__provider_proto, \"injectFault\", { configurable: true, writable: true, value: __f7 });\nvar __provider = Object.create(__provider_proto);\n\nfunction __f1() {\n return (function() {\n with({ }) {\n\nreturn function /*constructor*/() {\n };\n\n }\n }).apply(undefined, undefined).apply(this, arguments);\n}\n\nfunction __f3() {\n return (function() {\n with({ }) {\n\nreturn function (thisArg, _arguments, P, generator) {\n return new (P || (P = Promise))(function (resolve, reject) {\n function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }\n function rejected(value) { try { step(generator[\"throw\"](value)); } catch (e) { reject(e); } }\n function step(result) { result.done ? resolve(result.value) : new P(function (resolve) { resolve(result.value); }).then(fulfilled, rejected); }\n step((generator = generator.apply(thisArg, _arguments || [])).next());\n });\n};\n\n }\n }).apply(undefined, undefined).apply(this, arguments);\n}\n\nfunction __f2() {\n return (function() {\n with({ __awaiter: __f3 }) {\n\nreturn function /*diff*/(id, olds, news) {\n return __awaiter(this, void 0, void 0, function* () {\n let replaces = [];\n if (olds.replace !== news.replace) {\n replaces.push(\"replace\");\n }\n return {\n replaces: replaces,\n };\n });\n };\n\n }\n }).apply(undefined, undefined).apply(this, arguments);\n}\n\nfunction __f4() {\n return (function() {\n with({ __awaiter: __f3, currentID: 0 }) {\n\nreturn function /*create*/(inputs) {\n return __awaiter(this, void 0, void 0, function* () {\n if (this.inject) {\n throw this.inject;\n }\n return {\n id: (currentID++).toString(),\n outs: undefined,\n };\n });\n };\n\n }\n }).apply(undefined, undefined).apply(this, arguments);\n}\n\nfunction __f5() {\n return (function() {\n with({ __awaiter: __f3 }) {\n\nreturn function /*update*/(id, olds, news) {\n return __awaiter(this, void 0, void 0, function* () {\n if (this.inject) {\n throw this.inject;\n }\n return {};\n });\n };\n\n }\n }).apply(undefined, undefined).apply(this, arguments);\n}\n\nfunction __f6() {\n return (function() {\n with({ __awaiter: __f3 }) {\n\nreturn function /*delete*/(id, props) {\n return __awaiter(this, void 0, void 0, function* () {\n if (this.inject) {\n throw this.inject;\n }\n });\n };\n\n }\n }).apply(undefined, undefined).apply(this, arguments);\n}\n\nfunction __f7() {\n return (function() {\n with({ }) {\n\nreturn function /*injectFault*/(error) {\n this.inject = error;\n };\n\n }\n }).apply(undefined, undefined).apply(this, arguments);\n}\n\nfunction __f0() {\n return (function() {\n with({ provider: __provider }) {\n\nreturn () => provider;\n\n }\n }).apply(undefined, undefined).apply(this, arguments);\n}\n"</added>
|
|
<removed>- pulumi-nodejs:dynamic:Resource: (delete)
|
|
[id=0]
|
|
[urn=urn:pulumi:{{.StackName}}::steps::pulumi-nodejs:dynamic:Resource::d]
|
|
__provider: "exports.handler = __f0;\n\nvar __provider_proto = {};\n__f1.prototype = __provider_proto;\n__f1.instance = __provider;\nObject.defineProperty(__provider_proto, \"constructor\", { configurable: true, writable: true, value: __f1 });\nObject.defineProperty(__provider_proto, \"diff\", { configurable: true, writable: true, value: __f2 });\nObject.defineProperty(__provider_proto, \"create\", { configurable: true, writable: true, value: __f4 });\nObject.defineProperty(__provider_proto, \"update\", { configurable: true, writable: true, value: __f5 });\nObject.defineProperty(__provider_proto, \"delete\", { configurable: true, writable: true, value: __f6 });\nObject.defineProperty(__provider_proto, \"injectFault\", { configurable: true, writable: true, value: __f7 });\nvar __provider = Object.create(__provider_proto);\n\nfunction __f1() {\n return (function() {\n with({ }) {\n\nreturn function /*constructor*/() {\n };\n\n }\n }).apply(undefined, undefined).apply(this, arguments);\n}\n\nfunction __f3() {\n return (function() {\n with({ }) {\n\nreturn function (thisArg, _arguments, P, generator) {\n return new (P || (P = Promise))(function (resolve, reject) {\n function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }\n function rejected(value) { try { step(generator[\"throw\"](value)); } catch (e) { reject(e); } }\n function step(result) { result.done ? resolve(result.value) : new P(function (resolve) { resolve(result.value); }).then(fulfilled, rejected); }\n step((generator = generator.apply(thisArg, _arguments || [])).next());\n });\n};\n\n }\n }).apply(undefined, undefined).apply(this, arguments);\n}\n\nfunction __f2() {\n return (function() {\n with({ __awaiter: __f3 }) {\n\nreturn function /*diff*/(id, olds, news) {\n return __awaiter(this, void 0, void 0, function* () {\n let replaces = [];\n if (olds.replace !== news.replace) {\n replaces.push(\"replace\");\n }\n return {\n replaces: replaces,\n };\n });\n };\n\n }\n }).apply(undefined, undefined).apply(this, arguments);\n}\n\nfunction __f4() {\n return (function() {\n with({ __awaiter: __f3, currentID: 0 }) {\n\nreturn function /*create*/(inputs) {\n return __awaiter(this, void 0, void 0, function* () {\n if (this.inject) {\n throw this.inject;\n }\n return {\n id: (currentID++).toString(),\n outs: undefined,\n };\n });\n };\n\n }\n }).apply(undefined, undefined).apply(this, arguments);\n}\n\nfunction __f5() {\n return (function() {\n with({ __awaiter: __f3 }) {\n\nreturn function /*update*/(id, olds, news) {\n return __awaiter(this, void 0, void 0, function* () {\n if (this.inject) {\n throw this.inject;\n }\n return {};\n });\n };\n\n }\n }).apply(undefined, undefined).apply(this, arguments);\n}\n\nfunction __f6() {\n return (function() {\n with({ __awaiter: __f3 }) {\n\nreturn function /*delete*/(id, props) {\n return __awaiter(this, void 0, void 0, function* () {\n if (this.inject) {\n throw this.inject;\n }\n });\n };\n\n }\n }).apply(undefined, undefined).apply(this, arguments);\n}\n\nfunction __f7() {\n return (function() {\n with({ }) {\n\nreturn function /*injectFault*/(error) {\n this.inject = error;\n };\n\n }\n }).apply(undefined, undefined).apply(this, arguments);\n}\n\nfunction __f0() {\n return (function() {\n with({ provider: __provider }) {\n\nreturn () => provider;\n\n }\n }).apply(undefined, undefined).apply(this, arguments);\n}\n"</removed>
|
|
<info>info</info>: 2 changes performed:
|
|
<added>+ 1 resource created</added>
|
|
<removed>- 1 resource deleted</removed>
|
|
4 resources unchanged`, stack.StackName)
|
|
|
|
assertPreviewOutput(t, expected, buf.String())
|
|
|
|
buf.Reset()
|
|
},
|
|
},
|
|
{
|
|
Dir: "step3",
|
|
Additive: true,
|
|
Stdout: &buf,
|
|
Verbose: true,
|
|
ExtraRuntimeValidation: func(t *testing.T, stack integration.RuntimeValidationStackInfo) {
|
|
assert.NotNil(t, stack.Deployment)
|
|
assert.Equal(t, 4, len(stack.Deployment.Resources))
|
|
stackRes := stack.Deployment.Resources[0]
|
|
assert.Equal(t, resource.RootStackType, stackRes.URN.Type())
|
|
a := stack.Deployment.Resources[1]
|
|
assert.Equal(t, "a", string(a.URN.Name()))
|
|
c := stack.Deployment.Resources[2]
|
|
assert.Equal(t, "c", string(c.URN.Name()))
|
|
e := stack.Deployment.Resources[3]
|
|
assert.Equal(t, "e", string(e.URN.Name()))
|
|
|
|
expected :=
|
|
fillStackName(`<unchanged>Performing changes:
|
|
* pulumi:pulumi:Stack: (same)
|
|
[urn=urn:pulumi:{{.StackName}}::steps::pulumi:pulumi:Stack::steps-{{.StackName}}]</unchanged>
|
|
<removed>- pulumi-nodejs:dynamic:Resource: (delete)
|
|
[id=0]
|
|
[urn=urn:pulumi:{{.StackName}}::steps::pulumi-nodejs:dynamic:Resource::b]
|
|
__provider: "exports.handler = __f0;\n\nvar __provider_proto = {};\n__f1.prototype = __provider_proto;\n__f1.instance = __provider;\nObject.defineProperty(__provider_proto, \"constructor\", { configurable: true, writable: true, value: __f1 });\nObject.defineProperty(__provider_proto, \"diff\", { configurable: true, writable: true, value: __f2 });\nObject.defineProperty(__provider_proto, \"create\", { configurable: true, writable: true, value: __f4 });\nObject.defineProperty(__provider_proto, \"update\", { configurable: true, writable: true, value: __f5 });\nObject.defineProperty(__provider_proto, \"delete\", { configurable: true, writable: true, value: __f6 });\nObject.defineProperty(__provider_proto, \"injectFault\", { configurable: true, writable: true, value: __f7 });\nvar __provider = Object.create(__provider_proto);\n\nfunction __f1() {\n return (function() {\n with({ }) {\n\nreturn function /*constructor*/() {\n };\n\n }\n }).apply(undefined, undefined).apply(this, arguments);\n}\n\nfunction __f3() {\n return (function() {\n with({ }) {\n\nreturn function (thisArg, _arguments, P, generator) {\n return new (P || (P = Promise))(function (resolve, reject) {\n function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }\n function rejected(value) { try { step(generator[\"throw\"](value)); } catch (e) { reject(e); } }\n function step(result) { result.done ? resolve(result.value) : new P(function (resolve) { resolve(result.value); }).then(fulfilled, rejected); }\n step((generator = generator.apply(thisArg, _arguments || [])).next());\n });\n};\n\n }\n }).apply(undefined, undefined).apply(this, arguments);\n}\n\nfunction __f2() {\n return (function() {\n with({ __awaiter: __f3 }) {\n\nreturn function /*diff*/(id, olds, news) {\n return __awaiter(this, void 0, void 0, function* () {\n let replaces = [];\n if (olds.replace !== news.replace) {\n replaces.push(\"replace\");\n }\n return {\n replaces: replaces,\n };\n });\n };\n\n }\n }).apply(undefined, undefined).apply(this, arguments);\n}\n\nfunction __f4() {\n return (function() {\n with({ __awaiter: __f3, currentID: 0 }) {\n\nreturn function /*create*/(inputs) {\n return __awaiter(this, void 0, void 0, function* () {\n if (this.inject) {\n throw this.inject;\n }\n return {\n id: (currentID++).toString(),\n outs: undefined,\n };\n });\n };\n\n }\n }).apply(undefined, undefined).apply(this, arguments);\n}\n\nfunction __f5() {\n return (function() {\n with({ __awaiter: __f3 }) {\n\nreturn function /*update*/(id, olds, news) {\n return __awaiter(this, void 0, void 0, function* () {\n if (this.inject) {\n throw this.inject;\n }\n return {};\n });\n };\n\n }\n }).apply(undefined, undefined).apply(this, arguments);\n}\n\nfunction __f6() {\n return (function() {\n with({ __awaiter: __f3 }) {\n\nreturn function /*delete*/(id, props) {\n return __awaiter(this, void 0, void 0, function* () {\n if (this.inject) {\n throw this.inject;\n }\n });\n };\n\n }\n }).apply(undefined, undefined).apply(this, arguments);\n}\n\nfunction __f7() {\n return (function() {\n with({ }) {\n\nreturn function /*injectFault*/(error) {\n this.inject = error;\n };\n\n }\n }).apply(undefined, undefined).apply(this, arguments);\n}\n\nfunction __f0() {\n return (function() {\n with({ provider: __provider }) {\n\nreturn () => provider;\n\n }\n }).apply(undefined, undefined).apply(this, arguments);\n}\n"</removed>
|
|
<info>info</info>: 1 change performed:
|
|
<removed>- 1 resource deleted</removed>
|
|
4 resources unchanged`, stack.StackName)
|
|
|
|
assertPreviewOutput(t, expected, buf.String())
|
|
|
|
buf.Reset()
|
|
},
|
|
},
|
|
{
|
|
Dir: "step4",
|
|
Additive: true,
|
|
Stdout: &buf,
|
|
Verbose: true,
|
|
ExtraRuntimeValidation: func(t *testing.T, stack integration.RuntimeValidationStackInfo) {
|
|
assert.NotNil(t, stack.Deployment)
|
|
assert.Equal(t, 4, len(stack.Deployment.Resources))
|
|
stackRes := stack.Deployment.Resources[0]
|
|
assert.Equal(t, resource.RootStackType, stackRes.URN.Type())
|
|
a := stack.Deployment.Resources[1]
|
|
assert.Equal(t, "a", string(a.URN.Name()))
|
|
c := stack.Deployment.Resources[2]
|
|
assert.Equal(t, "c", string(c.URN.Name()))
|
|
e := stack.Deployment.Resources[3]
|
|
assert.Equal(t, "e", string(e.URN.Name()))
|
|
|
|
expected := fillStackName(`<unchanged>Performing changes:
|
|
* pulumi:pulumi:Stack: (same)
|
|
[urn=urn:pulumi:{{.StackName}}::steps::pulumi:pulumi:Stack::steps-{{.StackName}}]</unchanged>
|
|
<info>info</info>: no changes required:`, stack.StackName)
|
|
|
|
assertPreviewOutput(t, expected, buf.String())
|
|
|
|
buf.Reset()
|
|
},
|
|
},
|
|
{
|
|
Dir: "step5",
|
|
Additive: true,
|
|
Stdout: &buf,
|
|
Verbose: true,
|
|
ExtraRuntimeValidation: func(t *testing.T, stack integration.RuntimeValidationStackInfo) {
|
|
assert.NotNil(t, stack.Deployment)
|
|
assert.Equal(t, 1, len(stack.Deployment.Resources))
|
|
stackRes := stack.Deployment.Resources[0]
|
|
assert.Equal(t, resource.RootStackType, stackRes.URN.Type())
|
|
|
|
expected := fillStackName(`<unchanged>Performing changes:
|
|
* pulumi:pulumi:Stack: (same)
|
|
[urn=urn:pulumi:{{.StackName}}::steps::pulumi:pulumi:Stack::steps-{{.StackName}}]</unchanged>
|
|
<removed>- pulumi-nodejs:dynamic:Resource: (delete)
|
|
[id=0]
|
|
[urn=urn:pulumi:{{.StackName}}::steps::pulumi-nodejs:dynamic:Resource::e]
|
|
__provider: "exports.handler = __f0;\n\nvar __provider_proto = {};\n__f1.prototype = __provider_proto;\n__f1.instance = __provider;\nObject.defineProperty(__provider_proto, \"constructor\", { configurable: true, writable: true, value: __f1 });\nObject.defineProperty(__provider_proto, \"diff\", { configurable: true, writable: true, value: __f2 });\nObject.defineProperty(__provider_proto, \"create\", { configurable: true, writable: true, value: __f4 });\nObject.defineProperty(__provider_proto, \"update\", { configurable: true, writable: true, value: __f5 });\nObject.defineProperty(__provider_proto, \"delete\", { configurable: true, writable: true, value: __f6 });\nObject.defineProperty(__provider_proto, \"injectFault\", { configurable: true, writable: true, value: __f7 });\nvar __provider = Object.create(__provider_proto);\n\nfunction __f1() {\n return (function() {\n with({ }) {\n\nreturn function /*constructor*/() {\n };\n\n }\n }).apply(undefined, undefined).apply(this, arguments);\n}\n\nfunction __f3() {\n return (function() {\n with({ }) {\n\nreturn function (thisArg, _arguments, P, generator) {\n return new (P || (P = Promise))(function (resolve, reject) {\n function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }\n function rejected(value) { try { step(generator[\"throw\"](value)); } catch (e) { reject(e); } }\n function step(result) { result.done ? resolve(result.value) : new P(function (resolve) { resolve(result.value); }).then(fulfilled, rejected); }\n step((generator = generator.apply(thisArg, _arguments || [])).next());\n });\n};\n\n }\n }).apply(undefined, undefined).apply(this, arguments);\n}\n\nfunction __f2() {\n return (function() {\n with({ __awaiter: __f3 }) {\n\nreturn function /*diff*/(id, olds, news) {\n return __awaiter(this, void 0, void 0, function* () {\n let replaces = [];\n if (olds.replace !== news.replace) {\n replaces.push(\"replace\");\n }\n return {\n replaces: replaces,\n };\n });\n };\n\n }\n }).apply(undefined, undefined).apply(this, arguments);\n}\n\nfunction __f4() {\n return (function() {\n with({ __awaiter: __f3, currentID: 0 }) {\n\nreturn function /*create*/(inputs) {\n return __awaiter(this, void 0, void 0, function* () {\n if (this.inject) {\n throw this.inject;\n }\n return {\n id: (currentID++).toString(),\n outs: undefined,\n };\n });\n };\n\n }\n }).apply(undefined, undefined).apply(this, arguments);\n}\n\nfunction __f5() {\n return (function() {\n with({ __awaiter: __f3 }) {\n\nreturn function /*update*/(id, olds, news) {\n return __awaiter(this, void 0, void 0, function* () {\n if (this.inject) {\n throw this.inject;\n }\n return {};\n });\n };\n\n }\n }).apply(undefined, undefined).apply(this, arguments);\n}\n\nfunction __f6() {\n return (function() {\n with({ __awaiter: __f3 }) {\n\nreturn function /*delete*/(id, props) {\n return __awaiter(this, void 0, void 0, function* () {\n if (this.inject) {\n throw this.inject;\n }\n });\n };\n\n }\n }).apply(undefined, undefined).apply(this, arguments);\n}\n\nfunction __f7() {\n return (function() {\n with({ }) {\n\nreturn function /*injectFault*/(error) {\n this.inject = error;\n };\n\n }\n }).apply(undefined, undefined).apply(this, arguments);\n}\n\nfunction __f0() {\n return (function() {\n with({ provider: __provider }) {\n\nreturn () => provider;\n\n }\n }).apply(undefined, undefined).apply(this, arguments);\n}\n"</removed>
|
|
<removed>- pulumi-nodejs:dynamic:Resource: (delete)
|
|
[id=0]
|
|
[urn=urn:pulumi:{{.StackName}}::steps::pulumi-nodejs:dynamic:Resource::c]
|
|
__provider: "exports.handler = __f0;\n\nvar __provider_proto = {};\n__f1.prototype = __provider_proto;\n__f1.instance = __provider;\nObject.defineProperty(__provider_proto, \"constructor\", { configurable: true, writable: true, value: __f1 });\nObject.defineProperty(__provider_proto, \"diff\", { configurable: true, writable: true, value: __f2 });\nObject.defineProperty(__provider_proto, \"create\", { configurable: true, writable: true, value: __f4 });\nObject.defineProperty(__provider_proto, \"update\", { configurable: true, writable: true, value: __f5 });\nObject.defineProperty(__provider_proto, \"delete\", { configurable: true, writable: true, value: __f6 });\nObject.defineProperty(__provider_proto, \"injectFault\", { configurable: true, writable: true, value: __f7 });\nvar __provider = Object.create(__provider_proto);\n\nfunction __f1() {\n return (function() {\n with({ }) {\n\nreturn function /*constructor*/() {\n };\n\n }\n }).apply(undefined, undefined).apply(this, arguments);\n}\n\nfunction __f3() {\n return (function() {\n with({ }) {\n\nreturn function (thisArg, _arguments, P, generator) {\n return new (P || (P = Promise))(function (resolve, reject) {\n function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }\n function rejected(value) { try { step(generator[\"throw\"](value)); } catch (e) { reject(e); } }\n function step(result) { result.done ? resolve(result.value) : new P(function (resolve) { resolve(result.value); }).then(fulfilled, rejected); }\n step((generator = generator.apply(thisArg, _arguments || [])).next());\n });\n};\n\n }\n }).apply(undefined, undefined).apply(this, arguments);\n}\n\nfunction __f2() {\n return (function() {\n with({ __awaiter: __f3 }) {\n\nreturn function /*diff*/(id, olds, news) {\n return __awaiter(this, void 0, void 0, function* () {\n let replaces = [];\n if (olds.replace !== news.replace) {\n replaces.push(\"replace\");\n }\n return {\n replaces: replaces,\n };\n });\n };\n\n }\n }).apply(undefined, undefined).apply(this, arguments);\n}\n\nfunction __f4() {\n return (function() {\n with({ __awaiter: __f3, currentID: 0 }) {\n\nreturn function /*create*/(inputs) {\n return __awaiter(this, void 0, void 0, function* () {\n if (this.inject) {\n throw this.inject;\n }\n return {\n id: (currentID++).toString(),\n outs: undefined,\n };\n });\n };\n\n }\n }).apply(undefined, undefined).apply(this, arguments);\n}\n\nfunction __f5() {\n return (function() {\n with({ __awaiter: __f3 }) {\n\nreturn function /*update*/(id, olds, news) {\n return __awaiter(this, void 0, void 0, function* () {\n if (this.inject) {\n throw this.inject;\n }\n return {};\n });\n };\n\n }\n }).apply(undefined, undefined).apply(this, arguments);\n}\n\nfunction __f6() {\n return (function() {\n with({ __awaiter: __f3 }) {\n\nreturn function /*delete*/(id, props) {\n return __awaiter(this, void 0, void 0, function* () {\n if (this.inject) {\n throw this.inject;\n }\n });\n };\n\n }\n }).apply(undefined, undefined).apply(this, arguments);\n}\n\nfunction __f7() {\n return (function() {\n with({ }) {\n\nreturn function /*injectFault*/(error) {\n this.inject = error;\n };\n\n }\n }).apply(undefined, undefined).apply(this, arguments);\n}\n\nfunction __f0() {\n return (function() {\n with({ provider: __provider }) {\n\nreturn () => provider;\n\n }\n }).apply(undefined, undefined).apply(this, arguments);\n}\n"</removed>
|
|
<removed>- pulumi-nodejs:dynamic:Resource: (delete)
|
|
[id=0]
|
|
[urn=urn:pulumi:{{.StackName}}::steps::pulumi-nodejs:dynamic:Resource::a]
|
|
__provider: "exports.handler = __f0;\n\nvar __provider_proto = {};\n__f1.prototype = __provider_proto;\n__f1.instance = __provider;\nObject.defineProperty(__provider_proto, \"constructor\", { configurable: true, writable: true, value: __f1 });\nObject.defineProperty(__provider_proto, \"diff\", { configurable: true, writable: true, value: __f2 });\nObject.defineProperty(__provider_proto, \"create\", { configurable: true, writable: true, value: __f4 });\nObject.defineProperty(__provider_proto, \"update\", { configurable: true, writable: true, value: __f5 });\nObject.defineProperty(__provider_proto, \"delete\", { configurable: true, writable: true, value: __f6 });\nObject.defineProperty(__provider_proto, \"injectFault\", { configurable: true, writable: true, value: __f7 });\nvar __provider = Object.create(__provider_proto);\n\nfunction __f1() {\n return (function() {\n with({ }) {\n\nreturn function /*constructor*/() {\n };\n\n }\n }).apply(undefined, undefined).apply(this, arguments);\n}\n\nfunction __f3() {\n return (function() {\n with({ }) {\n\nreturn function (thisArg, _arguments, P, generator) {\n return new (P || (P = Promise))(function (resolve, reject) {\n function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }\n function rejected(value) { try { step(generator[\"throw\"](value)); } catch (e) { reject(e); } }\n function step(result) { result.done ? resolve(result.value) : new P(function (resolve) { resolve(result.value); }).then(fulfilled, rejected); }\n step((generator = generator.apply(thisArg, _arguments || [])).next());\n });\n};\n\n }\n }).apply(undefined, undefined).apply(this, arguments);\n}\n\nfunction __f2() {\n return (function() {\n with({ __awaiter: __f3 }) {\n\nreturn function /*diff*/(id, olds, news) {\n return __awaiter(this, void 0, void 0, function* () {\n let replaces = [];\n if (olds.replace !== news.replace) {\n replaces.push(\"replace\");\n }\n return {\n replaces: replaces,\n };\n });\n };\n\n }\n }).apply(undefined, undefined).apply(this, arguments);\n}\n\nfunction __f4() {\n return (function() {\n with({ __awaiter: __f3, currentID: 0 }) {\n\nreturn function /*create*/(inputs) {\n return __awaiter(this, void 0, void 0, function* () {\n if (this.inject) {\n throw this.inject;\n }\n return {\n id: (currentID++).toString(),\n outs: undefined,\n };\n });\n };\n\n }\n }).apply(undefined, undefined).apply(this, arguments);\n}\n\nfunction __f5() {\n return (function() {\n with({ __awaiter: __f3 }) {\n\nreturn function /*update*/(id, olds, news) {\n return __awaiter(this, void 0, void 0, function* () {\n if (this.inject) {\n throw this.inject;\n }\n return {};\n });\n };\n\n }\n }).apply(undefined, undefined).apply(this, arguments);\n}\n\nfunction __f6() {\n return (function() {\n with({ __awaiter: __f3 }) {\n\nreturn function /*delete*/(id, props) {\n return __awaiter(this, void 0, void 0, function* () {\n if (this.inject) {\n throw this.inject;\n }\n });\n };\n\n }\n }).apply(undefined, undefined).apply(this, arguments);\n}\n\nfunction __f7() {\n return (function() {\n with({ }) {\n\nreturn function /*injectFault*/(error) {\n this.inject = error;\n };\n\n }\n }).apply(undefined, undefined).apply(this, arguments);\n}\n\nfunction __f0() {\n return (function() {\n with({ provider: __provider }) {\n\nreturn () => provider;\n\n }\n }).apply(undefined, undefined).apply(this, arguments);\n}\n"</removed>
|
|
<info>info</info>: 3 changes performed:
|
|
<removed>- 3 resources deleted</removed>
|
|
1 resource unchanged`, stack.StackName)
|
|
|
|
assertPreviewOutput(t, expected, buf.String())
|
|
|
|
buf.Reset()
|
|
},
|
|
},
|
|
},
|
|
}
|
|
|
|
integration.ProgramTest(t, &opts)
|
|
}
|
|
|
|
func fillStackName(t string, stackName tokens.QName) string {
|
|
b := bytes.Buffer{}
|
|
template.Must(template.New("").Parse(t)).Execute(&b, struct{ StackName tokens.QName }{StackName: stackName})
|
|
return b.String()
|
|
}
|
|
|
|
func assertPreviewOutput(t *testing.T, expected, outputWithControlSeqeunces string) {
|
|
lines := strings.Split(outputWithControlSeqeunces, "\n")
|
|
|
|
// Remove lines from the output that differ across runs. The first two lines of the output are the command line
|
|
// we ran, the second is a message about updating the stack in the cloud, so we drop them.
|
|
lines = lines[2:]
|
|
|
|
// The last two lines include a call to stack export and a blank line. Drop them as well.
|
|
lines = lines[:len(lines)-2]
|
|
|
|
// If we are connected to a cloud who's URL scheme we recognize, the CLI prints a Permalink for the update, let's
|
|
// drop that (but only if it exists)
|
|
if strings.Index(lines[len(lines)-1], "Permalink: ") != -1 {
|
|
lines = lines[:len(lines)-1]
|
|
}
|
|
|
|
// Finally, we have information about how long the update took, which we also drop.
|
|
lines = lines[:len(lines)-1]
|
|
|
|
outputWithControlSeqeunces = strings.Join(lines, "\n")
|
|
assertProgramOutput(t, expected, outputWithControlSeqeunces)
|
|
}
|
|
|
|
func assertProgramOutput(t *testing.T, expected, outputWithControlSeqeunces string) {
|
|
// Now convert all the color control sequences over to a simpler form for test baseline purposes.
|
|
actual := convertControlSequences(outputWithControlSeqeunces)
|
|
|
|
if expected != actual {
|
|
diff, _ := difflib.GetUnifiedDiffString(difflib.UnifiedDiff{
|
|
A: difflib.SplitLines(expected),
|
|
B: difflib.SplitLines(actual),
|
|
FromFile: "Expected",
|
|
FromDate: "",
|
|
ToFile: "Actual",
|
|
ToDate: "",
|
|
Context: 0,
|
|
})
|
|
|
|
assert.Fail(t, "Difference between expected and actual:\n"+diff)
|
|
}
|
|
}
|
|
|
|
type ColorEnum string
|
|
|
|
const (
|
|
Clear ColorEnum = "clear"
|
|
Unchanged ColorEnum = "unchanged"
|
|
Added ColorEnum = "added"
|
|
Removed ColorEnum = "removed"
|
|
Info ColorEnum = "info"
|
|
Debug ColorEnum = "debug"
|
|
)
|
|
|
|
// convertControlSequences takes in the output of the pulumi update command (including color sequences)
|
|
// and converts it to a simpler form that is easier to baseline. Control sequences,
|
|
// like '<{%fg 2%}}', are converted to simpler code like <added>, with reset controls closing those
|
|
// tags.
|
|
//
|
|
// It's a lot of string munging, but makes it much easier to baseline and validate update diffs.
|
|
func convertControlSequences(text string) string {
|
|
getColor := func(startInclusive, endExclusive int) ColorEnum {
|
|
switch text[startInclusive:endExclusive] {
|
|
case "<{%reset%}>":
|
|
return Clear
|
|
case "<{%fg 1%}>":
|
|
return Removed
|
|
case "<{%fg 2%}>":
|
|
return Added
|
|
case "<{%fg 5%}>":
|
|
return Info
|
|
case "<{%fg 8%}>":
|
|
return Unchanged
|
|
case "<{%fg 7%}>":
|
|
return Debug
|
|
default:
|
|
panic("Unexpected match: " + text[startInclusive:endExclusive])
|
|
}
|
|
}
|
|
|
|
allWhitespace := func(startInclusive, endExclusive int) bool {
|
|
for i := startInclusive; i < endExclusive; i++ {
|
|
if !unicode.IsSpace(rune(text[i])) {
|
|
return false
|
|
}
|
|
}
|
|
|
|
return true
|
|
}
|
|
|
|
// Normalize all \r\n to \n's. it makes all the string processing we need to do much simpler.
|
|
text = strings.Replace(text, "\r\n", "\n", -1)
|
|
|
|
var result bytes.Buffer
|
|
currentColor := Clear
|
|
index := 0
|
|
|
|
anyTagRegex := regexp.MustCompile(`<\{.*?\}>`)
|
|
allTagStartEndPairs := anyTagRegex.FindAllStringIndex(text, -1)
|
|
|
|
for pairIndex, startEndPair := range allTagStartEndPairs {
|
|
startInclusive := startEndPair[0]
|
|
endExclusive := startEndPair[1]
|
|
|
|
if startInclusive > index {
|
|
result.WriteString(text[index:startInclusive])
|
|
}
|
|
|
|
nextColor := getColor(startInclusive, endExclusive)
|
|
|
|
index = endExclusive
|
|
if nextColor == currentColor {
|
|
// Ignore it if we see two of the same color in a row.
|
|
continue
|
|
}
|
|
|
|
if nextColor == Clear {
|
|
// ignore a clear if it's just followed by whitespace, and then a switch back to
|
|
// our current color. i.e. something of the form: <Add> ... <Clear> whitespace <Add> ...
|
|
if pairIndex+1 < len(allTagStartEndPairs) {
|
|
nextNextPair := allTagStartEndPairs[pairIndex+1]
|
|
nextNextColor := getColor(nextNextPair[0], nextNextPair[1])
|
|
|
|
if nextNextColor == currentColor {
|
|
if allWhitespace(endExclusive, nextNextPair[0]) {
|
|
continue
|
|
}
|
|
}
|
|
}
|
|
|
|
result.WriteString("</" + string(currentColor) + ">")
|
|
} else {
|
|
result.WriteString("<" + string(nextColor) + ">")
|
|
}
|
|
|
|
currentColor = nextColor
|
|
}
|
|
|
|
if index < len(text) {
|
|
result.WriteString(text[index:])
|
|
}
|
|
|
|
if currentColor != Clear {
|
|
result.WriteString("</" + string(currentColor) + ">")
|
|
}
|
|
|
|
taggedString := result.String()
|
|
|
|
// We'll routinely end up with a line, followed by a newline, followed by and endtag (due to
|
|
// reset chars being written after lines are written). To make this cleaner in the baseline
|
|
// swap the two so the line ends with the endtag and then is followed by the newline.s
|
|
r, _ := regexp.Compile(`(\n)(\<\/[a-z]+\>)`)
|
|
replacedString := r.ReplaceAllString(taggedString, "$2$1")
|
|
|
|
return replacedString
|
|
}
|