mirror of https://github.com/pulumi/pulumi.git
177 lines
6.0 KiB
Go
177 lines
6.0 KiB
Go
// Copyright 2019-2024, 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 engine
|
|
|
|
import (
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/resource"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/resource/plugin"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/util/contract"
|
|
)
|
|
|
|
// getProperty fetches the child property with the indicated key from the given property value. If the key does not
|
|
// exist, it returns an empty `PropertyValue`.
|
|
func getProperty(key interface{}, v resource.PropertyValue) resource.PropertyValue {
|
|
switch {
|
|
case v.IsArray():
|
|
index, ok := key.(int)
|
|
if !ok || index < 0 || index >= len(v.ArrayValue()) {
|
|
return resource.PropertyValue{}
|
|
}
|
|
return v.ArrayValue()[index]
|
|
case v.IsObject():
|
|
k, ok := key.(string)
|
|
if !ok {
|
|
return resource.PropertyValue{}
|
|
}
|
|
return v.ObjectValue()[resource.PropertyKey(k)]
|
|
case v.IsComputed() || v.IsOutput() || v.IsSecret():
|
|
// We consider the contents of these values opaque and return them as-is, as we cannot know whether or not the
|
|
// value will or does contain an element with the given key.
|
|
return v
|
|
default:
|
|
return resource.PropertyValue{}
|
|
}
|
|
}
|
|
|
|
// addDiff inserts a diff of the given kind at the given path into the parent ValueDiff.
|
|
//
|
|
// If the path consists of a single element, a diff of the indicated kind is inserted directly. Otherwise, if the
|
|
// property named by the first element of the path exists in both parents, we snip off the first element of the path
|
|
// and recurse into the property itself. If the property does not exist in one parent or the other, the diff kind is
|
|
// disregarded and the change is treated as either an Add or a Delete.
|
|
func addDiff(path resource.PropertyPath, kind plugin.DiffKind, parent *resource.ValueDiff,
|
|
oldParent, newParent resource.PropertyValue,
|
|
) {
|
|
contract.Requiref(len(path) > 0, "path", "must not be empty")
|
|
|
|
element := path[0]
|
|
|
|
old, new := getProperty(element, oldParent), getProperty(element, newParent)
|
|
|
|
switch element := element.(type) {
|
|
case int:
|
|
if parent.Array == nil {
|
|
parent.Array = &resource.ArrayDiff{
|
|
Adds: make(map[int]resource.PropertyValue),
|
|
Deletes: make(map[int]resource.PropertyValue),
|
|
Sames: make(map[int]resource.PropertyValue),
|
|
Updates: make(map[int]resource.ValueDiff),
|
|
}
|
|
}
|
|
|
|
// For leaf diffs, the provider tells us exactly what to record. For other diffs, we will derive the
|
|
// difference from the old and new property values.
|
|
if len(path) == 1 {
|
|
switch kind {
|
|
case plugin.DiffAdd, plugin.DiffAddReplace:
|
|
parent.Array.Adds[element] = new
|
|
case plugin.DiffDelete, plugin.DiffDeleteReplace:
|
|
parent.Array.Deletes[element] = old
|
|
case plugin.DiffUpdate, plugin.DiffUpdateReplace:
|
|
valueDiff := resource.ValueDiff{Old: old, New: new}
|
|
if d := old.Diff(new); d != nil {
|
|
valueDiff = *d
|
|
}
|
|
parent.Array.Updates[element] = valueDiff
|
|
default:
|
|
contract.Failf("unexpected diff kind %v", kind)
|
|
}
|
|
} else {
|
|
switch {
|
|
case old.IsNull() && !new.IsNull():
|
|
parent.Array.Adds[element] = new
|
|
case !old.IsNull() && new.IsNull():
|
|
parent.Array.Deletes[element] = old
|
|
default:
|
|
ed := parent.Array.Updates[element]
|
|
addDiff(path[1:], kind, &ed, old, new)
|
|
parent.Array.Updates[element] = ed
|
|
}
|
|
}
|
|
case string:
|
|
if parent.Object == nil {
|
|
parent.Object = &resource.ObjectDiff{
|
|
Adds: make(resource.PropertyMap),
|
|
Deletes: make(resource.PropertyMap),
|
|
Sames: make(resource.PropertyMap),
|
|
Updates: make(map[resource.PropertyKey]resource.ValueDiff),
|
|
}
|
|
}
|
|
|
|
e := resource.PropertyKey(element)
|
|
if len(path) == 1 {
|
|
switch kind {
|
|
case plugin.DiffAdd, plugin.DiffAddReplace:
|
|
parent.Object.Adds[e] = new
|
|
case plugin.DiffDelete, plugin.DiffDeleteReplace:
|
|
parent.Object.Deletes[e] = old
|
|
case plugin.DiffUpdate, plugin.DiffUpdateReplace:
|
|
valueDiff := resource.ValueDiff{Old: old, New: new}
|
|
if d := old.Diff(new); d != nil {
|
|
valueDiff = *d
|
|
}
|
|
parent.Object.Updates[e] = valueDiff
|
|
default:
|
|
contract.Failf("unexpected diff kind %v", kind)
|
|
}
|
|
} else {
|
|
switch {
|
|
case old.IsNull() && !new.IsNull():
|
|
parent.Object.Adds[e] = new
|
|
case !old.IsNull() && new.IsNull():
|
|
parent.Object.Deletes[e] = old
|
|
default:
|
|
ed := parent.Object.Updates[e]
|
|
addDiff(path[1:], kind, &ed, old, new)
|
|
parent.Object.Updates[e] = ed
|
|
}
|
|
}
|
|
default:
|
|
contract.Failf("unexpected path element type: %T", element)
|
|
}
|
|
}
|
|
|
|
// TranslateDetailedDiff converts the detailed diff stored in the step event into an ObjectDiff that is appropriate
|
|
// for display.
|
|
func TranslateDetailedDiff(step *StepEventMetadata, refresh bool) *resource.ObjectDiff {
|
|
contract.Assertf(step.DetailedDiff != nil, "%v step has no detailed diff", step.Op)
|
|
|
|
// The rich diff is presented as a list of simple JS property paths and corresponding diffs. We translate this to
|
|
// an ObjectDiff by iterating the list and inserting ValueDiffs that reflect the changes in the detailed diff. Old
|
|
// values are always taken from a step's Outputs; new values are always taken from its Inputs.
|
|
|
|
var diff resource.ValueDiff
|
|
for path, pdiff := range step.DetailedDiff {
|
|
elements, err := resource.ParsePropertyPath(path)
|
|
if err != nil {
|
|
elements = []interface{}{path}
|
|
}
|
|
|
|
olds := resource.NewObjectProperty(step.Old.Outputs)
|
|
if pdiff.InputDiff {
|
|
olds = resource.NewObjectProperty(step.Old.Inputs)
|
|
}
|
|
|
|
news := resource.NewObjectProperty(step.New.Inputs)
|
|
if refresh {
|
|
news = resource.NewObjectProperty(step.New.Outputs)
|
|
}
|
|
|
|
addDiff(elements, pdiff.Kind, &diff, olds, news)
|
|
}
|
|
|
|
return diff.Object
|
|
}
|