opentofu/internal/command/jsonformat/differ/list.go

181 lines
5.0 KiB
Go
Raw Normal View History

package differ
import (
"reflect"
"github.com/zclconf/go-cty/cty"
"github.com/hashicorp/terraform/internal/command/jsonformat/change"
"github.com/hashicorp/terraform/internal/command/jsonprovider"
"github.com/hashicorp/terraform/internal/plans"
)
func (v Value) computeAttributeChangeAsList(elementType cty.Type) change.Change {
var elements []change.Change
current := v.getDefaultActionForIteration()
v.processList(elementType.IsObjectType(), func(value Value) {
element := value.computeChangeForType(elementType)
elements = append(elements, element)
current = compareActions(current, element.Action())
})
return change.New(change.List(elements), current, v.replacePath())
}
func (v Value) computeAttributeChangeAsNestedList(attributes map[string]*jsonprovider.Attribute) change.Change {
var elements []change.Change
current := v.getDefaultActionForIteration()
v.processNestedList(func(value Value) {
element := value.computeChangeForNestedAttribute(&jsonprovider.NestedType{
Attributes: attributes,
NestingMode: "single",
})
elements = append(elements, element)
current = compareActions(current, element.Action())
})
return change.New(change.NestedList(elements), current, v.replacePath())
}
func (v Value) computeBlockChangesAsList(block *jsonprovider.Block) ([]change.Change, plans.Action) {
var elements []change.Change
current := v.getDefaultActionForIteration()
v.processNestedList(func(value Value) {
element := value.ComputeChangeForBlock(block)
elements = append(elements, element)
current = compareActions(current, element.Action())
})
return elements, current
}
func (v Value) processNestedList(process func(value Value)) {
sliceValue := v.asSlice()
for ix := 0; ix < len(sliceValue.Before) || ix < len(sliceValue.After); ix++ {
process(sliceValue.getChild(ix, ix, false))
}
}
func (v Value) processList(isObjType bool, process func(value Value)) {
sliceValue := v.asSlice()
lcs := lcs(sliceValue.Before, sliceValue.After)
var beforeIx, afterIx, lcsIx int
for beforeIx < len(sliceValue.Before) || afterIx < len(sliceValue.After) || lcsIx < len(lcs) {
// Step through all the before values until we hit the next item in the
// longest common subsequence. We are going to just say that all of
// these have been deleted.
for beforeIx < len(sliceValue.Before) && (lcsIx >= len(lcs) || !reflect.DeepEqual(sliceValue.Before[beforeIx], lcs[lcsIx])) {
isObjectDiff := isObjType && afterIx < len(sliceValue.After) && (lcsIx >= len(lcs) || !reflect.DeepEqual(sliceValue.After[afterIx], lcs[lcsIx]))
if isObjectDiff {
process(sliceValue.getChild(beforeIx, afterIx, false))
beforeIx++
afterIx++
continue
}
process(sliceValue.getChild(beforeIx, len(sliceValue.After), false))
beforeIx++
}
// Now, step through all the after values until hit the next item in the
// LCS. We are going to say that all of these have been created.
for afterIx < len(sliceValue.After) && (lcsIx >= len(lcs) || !reflect.DeepEqual(sliceValue.After[afterIx], lcs[lcsIx])) {
process(sliceValue.getChild(len(sliceValue.Before), afterIx, false))
afterIx++
}
// Finally, add the item in common as unchanged.
if lcsIx < len(lcs) {
process(sliceValue.getChild(beforeIx, afterIx, false))
beforeIx++
afterIx++
lcsIx++
}
}
}
func lcs(xs, ys []interface{}) []interface{} {
if len(xs) == 0 || len(ys) == 0 {
return make([]interface{}, 0)
}
c := make([]int, len(xs)*len(ys))
eqs := make([]bool, len(xs)*len(ys))
w := len(xs)
for y := 0; y < len(ys); y++ {
for x := 0; x < len(xs); x++ {
eq := false
if reflect.DeepEqual(xs[x], ys[y]) {
eq = true
eqs[(w*y)+x] = true // equality tests can be expensive, so cache it
}
if eq {
// Sequence gets one longer than for the cell at top left,
// since we'd append a new item to the sequence here.
if x == 0 || y == 0 {
c[(w*y)+x] = 1
} else {
c[(w*y)+x] = c[(w*(y-1))+(x-1)] + 1
}
} else {
// We follow the longest of the sequence above and the sequence
// to the left of us in the matrix.
l := 0
u := 0
if x > 0 {
l = c[(w*y)+(x-1)]
}
if y > 0 {
u = c[(w*(y-1))+x]
}
if l > u {
c[(w*y)+x] = l
} else {
c[(w*y)+x] = u
}
}
}
}
// The bottom right cell tells us how long our longest sequence will be
seq := make([]interface{}, c[len(c)-1])
// Now we will walk back from the bottom right cell, finding again all
// of the equal pairs to construct our sequence.
x := len(xs) - 1
y := len(ys) - 1
i := len(seq) - 1
for x > -1 && y > -1 {
if eqs[(w*y)+x] {
// Add the value to our result list and then walk diagonally
// up and to the left.
seq[i] = xs[x]
x--
y--
i--
} else {
// Take the path with the greatest sequence length in the matrix.
l := 0
u := 0
if x > 0 {
l = c[(w*y)+(x-1)]
}
if y > 0 {
u = c[(w*(y-1))+x]
}
if l > u {
x--
} else {
y--
}
}
}
if i > -1 {
// should never happen if the matrix was constructed properly
panic("not enough elements in sequence")
}
return seq
}