2018-06-07 19:27:57 -05:00
|
|
|
package states
|
|
|
|
|
|
|
|
import (
|
|
|
|
"reflect"
|
2021-05-05 19:35:25 -05:00
|
|
|
|
2021-05-17 14:00:50 -05:00
|
|
|
"github.com/hashicorp/terraform/internal/addrs"
|
2018-06-07 19:27:57 -05:00
|
|
|
)
|
|
|
|
|
|
|
|
// Equal returns true if the receiver is functionally equivalent to other,
|
|
|
|
// including any ephemeral portions of the state that would not be included
|
|
|
|
// if the state were saved to files.
|
|
|
|
//
|
|
|
|
// To test only the persistent portions of two states for equality, instead
|
|
|
|
// use statefile.StatesMarshalEqual.
|
|
|
|
func (s *State) Equal(other *State) bool {
|
|
|
|
// For the moment this is sufficient, but we may need to do something
|
|
|
|
// more elaborate in future if we have any portions of state that require
|
|
|
|
// more sophisticated comparisons.
|
|
|
|
return reflect.DeepEqual(s, other)
|
|
|
|
}
|
2021-05-05 19:35:25 -05:00
|
|
|
|
|
|
|
// ManagedResourcesEqual returns true if all of the managed resources tracked
|
|
|
|
// in the reciever are functionally equivalent to the same tracked in the
|
|
|
|
// other given state.
|
|
|
|
//
|
|
|
|
// This is a more constrained version of Equal that disregards other
|
|
|
|
// differences, including but not limited to changes to data resources and
|
|
|
|
// changes to output values.
|
|
|
|
func (s *State) ManagedResourcesEqual(other *State) bool {
|
|
|
|
// First, some accommodations for situations where one of the objects is
|
|
|
|
// nil, for robustness since we sometimes use a nil state to represent
|
|
|
|
// a prior state being entirely absent.
|
|
|
|
if s == nil && other == nil {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
if s == nil {
|
2021-10-13 14:21:23 -05:00
|
|
|
return !other.HasManagedResourceInstanceObjects()
|
2021-05-05 19:35:25 -05:00
|
|
|
}
|
|
|
|
if other == nil {
|
2021-10-13 14:21:23 -05:00
|
|
|
return !s.HasManagedResourceInstanceObjects()
|
2021-05-05 19:35:25 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
// If we get here then both states are non-nil.
|
|
|
|
|
|
|
|
// sameManagedResources tests that its second argument has all the
|
|
|
|
// resources that the first one does, so we'll call it twice with the
|
|
|
|
// arguments inverted to ensure that we'll also catch situations where
|
|
|
|
// the second has resources that the first does not.
|
|
|
|
return sameManagedResources(s, other) && sameManagedResources(other, s)
|
|
|
|
}
|
|
|
|
|
|
|
|
func sameManagedResources(s1, s2 *State) bool {
|
|
|
|
for _, ms := range s1.Modules {
|
|
|
|
for _, rs := range ms.Resources {
|
|
|
|
addr := rs.Addr
|
|
|
|
if addr.Resource.Mode != addrs.ManagedResourceMode {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
otherRS := s2.Resource(addr)
|
|
|
|
if !reflect.DeepEqual(rs, otherRS) {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true
|
|
|
|
|
|
|
|
}
|