mirror of
https://github.com/opentofu/opentofu.git
synced 2025-01-04 13:17:43 -06:00
204 lines
4.2 KiB
Go
204 lines
4.2 KiB
Go
package terraform
|
|
|
|
import (
|
|
"bytes"
|
|
"encoding/gob"
|
|
"errors"
|
|
"fmt"
|
|
"io"
|
|
"sort"
|
|
"strings"
|
|
"sync"
|
|
)
|
|
|
|
// The format byte is prefixed into the diff file format so that we have
|
|
// the ability in the future to change the file format if we want for any
|
|
// reason.
|
|
const diffFormatByte byte = 1
|
|
|
|
// Diff tracks the differences between resources to apply.
|
|
type Diff struct {
|
|
Resources map[string]*ResourceDiff
|
|
once sync.Once
|
|
}
|
|
|
|
// ReadDiff reads a diff structure out of a reader in the format that
|
|
// was written by WriteDiff.
|
|
func ReadDiff(src io.Reader) (*Diff, error) {
|
|
var result *Diff
|
|
|
|
var formatByte [1]byte
|
|
n, err := src.Read(formatByte[:])
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if n != len(formatByte) {
|
|
return nil, errors.New("failed to read diff version byte")
|
|
}
|
|
|
|
if formatByte[0] != diffFormatByte {
|
|
return nil, fmt.Errorf("unknown diff file version: %d", formatByte[0])
|
|
}
|
|
|
|
dec := gob.NewDecoder(src)
|
|
if err := dec.Decode(&result); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return result, nil
|
|
}
|
|
|
|
// WriteDiff writes a diff somewhere in a binary format.
|
|
func WriteDiff(d *Diff, dst io.Writer) error {
|
|
n, err := dst.Write([]byte{diffFormatByte})
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if n != 1 {
|
|
return errors.New("failed to write diff version byte")
|
|
}
|
|
|
|
return gob.NewEncoder(dst).Encode(d)
|
|
}
|
|
|
|
func (d *Diff) init() {
|
|
d.once.Do(func() {
|
|
if d.Resources == nil {
|
|
d.Resources = make(map[string]*ResourceDiff)
|
|
}
|
|
})
|
|
}
|
|
|
|
// Empty returns true if the diff has no changes.
|
|
func (d *Diff) Empty() bool {
|
|
if len(d.Resources) == 0 {
|
|
return true
|
|
}
|
|
|
|
for _, rd := range d.Resources {
|
|
if !rd.Empty() {
|
|
return false
|
|
}
|
|
}
|
|
|
|
return true
|
|
}
|
|
|
|
// String outputs the diff in a long but command-line friendly output
|
|
// format that users can read to quickly inspect a diff.
|
|
func (d *Diff) String() string {
|
|
var buf bytes.Buffer
|
|
|
|
names := make([]string, 0, len(d.Resources))
|
|
for name, _ := range d.Resources {
|
|
names = append(names, name)
|
|
}
|
|
sort.Strings(names)
|
|
|
|
for _, name := range names {
|
|
rdiff := d.Resources[name]
|
|
|
|
crud := "UPDATE"
|
|
if rdiff.RequiresNew() && rdiff.Destroy {
|
|
crud = "DESTROY/CREATE"
|
|
} else if rdiff.Destroy {
|
|
crud = "DESTROY"
|
|
} else if rdiff.RequiresNew() {
|
|
crud = "CREATE"
|
|
}
|
|
|
|
buf.WriteString(fmt.Sprintf(
|
|
"%s: %s\n",
|
|
crud,
|
|
name))
|
|
|
|
keyLen := 0
|
|
keys := make([]string, 0, len(rdiff.Attributes))
|
|
for key, _ := range rdiff.Attributes {
|
|
keys = append(keys, key)
|
|
if len(key) > keyLen {
|
|
keyLen = len(key)
|
|
}
|
|
}
|
|
sort.Strings(keys)
|
|
|
|
for _, attrK := range keys {
|
|
attrDiff := rdiff.Attributes[attrK]
|
|
|
|
v := attrDiff.New
|
|
if attrDiff.NewComputed {
|
|
v = "<computed>"
|
|
}
|
|
|
|
newResource := ""
|
|
if attrDiff.RequiresNew {
|
|
newResource = " (forces new resource)"
|
|
}
|
|
|
|
buf.WriteString(fmt.Sprintf(
|
|
" %s:%s %#v => %#v%s\n",
|
|
attrK,
|
|
strings.Repeat(" ", keyLen-len(attrK)),
|
|
attrDiff.Old,
|
|
v,
|
|
newResource))
|
|
}
|
|
}
|
|
|
|
return buf.String()
|
|
}
|
|
|
|
// ResourceDiff is the diff of a resource from some state to another.
|
|
type ResourceDiff struct {
|
|
Attributes map[string]*ResourceAttrDiff
|
|
Destroy bool
|
|
}
|
|
|
|
// ResourceAttrDiff is the diff of a single attribute of a resource.
|
|
type ResourceAttrDiff struct {
|
|
Old string // Old Value
|
|
New string // New Value
|
|
NewComputed bool // True if new value is computed (unknown currently)
|
|
RequiresNew bool // True if change requires new resource
|
|
Type DiffAttrType
|
|
}
|
|
|
|
// DiffAttrType is an enum type that says whether a resource attribute
|
|
// diff is an input attribute (comes from the configuration) or an
|
|
// output attribute (comes as a result of applying the configuration). An
|
|
// example input would be "ami" for AWS and an example output would be
|
|
// "private_ip".
|
|
type DiffAttrType byte
|
|
|
|
const (
|
|
DiffAttrUnknown DiffAttrType = iota
|
|
DiffAttrInput
|
|
DiffAttrOutput
|
|
)
|
|
|
|
// Empty returns true if this diff encapsulates no changes.
|
|
// TODO(mitchellh): test
|
|
func (d *ResourceDiff) Empty() bool {
|
|
if d == nil {
|
|
return true
|
|
}
|
|
|
|
return !d.Destroy && len(d.Attributes) == 0
|
|
}
|
|
|
|
// RequiresNew returns true if the diff requires the creation of a new
|
|
// resource (implying the destruction of the old).
|
|
func (d *ResourceDiff) RequiresNew() bool {
|
|
if d == nil {
|
|
return false
|
|
}
|
|
|
|
for _, rd := range d.Attributes {
|
|
if rd.RequiresNew {
|
|
return true
|
|
}
|
|
}
|
|
|
|
return false
|
|
}
|