mirror of
https://github.com/opentofu/opentofu.git
synced 2024-12-28 18:01:01 -06:00
f2adfb6e2a
In many ways a deposed object is equivalent to an orphaned current object in that the only action we can take with it is to destroy it. However, we do still need to take some preparation steps in both cases: first, we must ensure we track the upgraded version of the existing object so that we'll be able to successfully render our plan, and secondly we must refresh the existing object to make sure it still exists in the remote system. We were previously doing these extra steps for orphan objects but not for deposed ones, which meant that the behavior for deposed objects would be subtly different and violate the invariants our callers expect in order to display a plan. This also created the risk that a deposed object already deleted in the remote system would become "stuck" because Terraform would still plan to destroy it, which might cause the provider to return an error when it tries to delete an already-absent object. This also makes the deposed object planning take into account the "skipPlanChanges" flag, which is important to get a correct result in the "refresh only" planning mode. It's a shame that we have almost identical code handling both the orphan and deposed situations, but they differ in that the latter must call different functions to interact with the deposed rather than the current objects in the state. Perhaps a later change can improve on this with some more refactoring, but this commit is already a little more disruptive than I'd like and so I'm intentionally deferring that for another day.
362 lines
8.9 KiB
Go
362 lines
8.9 KiB
Go
package views
|
|
|
|
import (
|
|
"bufio"
|
|
"bytes"
|
|
"fmt"
|
|
"strings"
|
|
"sync"
|
|
"time"
|
|
"unicode"
|
|
|
|
"github.com/zclconf/go-cty/cty"
|
|
|
|
"github.com/hashicorp/terraform/addrs"
|
|
"github.com/hashicorp/terraform/command/format"
|
|
"github.com/hashicorp/terraform/plans"
|
|
"github.com/hashicorp/terraform/providers"
|
|
"github.com/hashicorp/terraform/states"
|
|
"github.com/hashicorp/terraform/terraform"
|
|
)
|
|
|
|
const defaultPeriodicUiTimer = 10 * time.Second
|
|
const maxIdLen = 80
|
|
|
|
func NewUiHook(view *View) *UiHook {
|
|
return &UiHook{
|
|
view: view,
|
|
periodicUiTimer: defaultPeriodicUiTimer,
|
|
resources: make(map[string]uiResourceState),
|
|
}
|
|
}
|
|
|
|
type UiHook struct {
|
|
terraform.NilHook
|
|
|
|
view *View
|
|
viewLock sync.Mutex
|
|
|
|
periodicUiTimer time.Duration
|
|
|
|
resources map[string]uiResourceState
|
|
resourcesLock sync.Mutex
|
|
}
|
|
|
|
var _ terraform.Hook = (*UiHook)(nil)
|
|
|
|
// uiResourceState tracks the state of a single resource
|
|
type uiResourceState struct {
|
|
DispAddr string
|
|
IDKey, IDValue string
|
|
Op uiResourceOp
|
|
Start time.Time
|
|
|
|
DoneCh chan struct{} // To be used for cancellation
|
|
|
|
done chan struct{} // used to coordinate tests
|
|
}
|
|
|
|
// uiResourceOp is an enum for operations on a resource
|
|
type uiResourceOp byte
|
|
|
|
const (
|
|
uiResourceUnknown uiResourceOp = iota
|
|
uiResourceCreate
|
|
uiResourceModify
|
|
uiResourceDestroy
|
|
uiResourceRead
|
|
)
|
|
|
|
func (h *UiHook) PreApply(addr addrs.AbsResourceInstance, gen states.Generation, action plans.Action, priorState, plannedNewState cty.Value) (terraform.HookAction, error) {
|
|
dispAddr := addr.String()
|
|
if gen != states.CurrentGen {
|
|
dispAddr = fmt.Sprintf("%s (deposed object %s)", dispAddr, gen)
|
|
}
|
|
|
|
var operation string
|
|
var op uiResourceOp
|
|
idKey, idValue := format.ObjectValueIDOrName(priorState)
|
|
switch action {
|
|
case plans.Delete:
|
|
operation = "Destroying..."
|
|
op = uiResourceDestroy
|
|
case plans.Create:
|
|
operation = "Creating..."
|
|
op = uiResourceCreate
|
|
case plans.Update:
|
|
operation = "Modifying..."
|
|
op = uiResourceModify
|
|
case plans.Read:
|
|
operation = "Reading..."
|
|
op = uiResourceRead
|
|
default:
|
|
// We don't expect any other actions in here, so anything else is a
|
|
// bug in the caller but we'll ignore it in order to be robust.
|
|
h.println(fmt.Sprintf("(Unknown action %s for %s)", action, dispAddr))
|
|
return terraform.HookActionContinue, nil
|
|
}
|
|
|
|
var stateIdSuffix string
|
|
if idKey != "" && idValue != "" {
|
|
stateIdSuffix = fmt.Sprintf(" [%s=%s]", idKey, idValue)
|
|
} else {
|
|
// Make sure they are both empty so we can deal with this more
|
|
// easily in the other hook methods.
|
|
idKey = ""
|
|
idValue = ""
|
|
}
|
|
|
|
h.println(fmt.Sprintf(
|
|
h.view.colorize.Color("[reset][bold]%s: %s%s[reset]"),
|
|
dispAddr,
|
|
operation,
|
|
stateIdSuffix,
|
|
))
|
|
|
|
key := addr.String()
|
|
uiState := uiResourceState{
|
|
DispAddr: key,
|
|
IDKey: idKey,
|
|
IDValue: idValue,
|
|
Op: op,
|
|
Start: time.Now().Round(time.Second),
|
|
DoneCh: make(chan struct{}),
|
|
done: make(chan struct{}),
|
|
}
|
|
|
|
h.resourcesLock.Lock()
|
|
h.resources[key] = uiState
|
|
h.resourcesLock.Unlock()
|
|
|
|
// Start goroutine that shows progress
|
|
go h.stillApplying(uiState)
|
|
|
|
return terraform.HookActionContinue, nil
|
|
}
|
|
|
|
func (h *UiHook) stillApplying(state uiResourceState) {
|
|
defer close(state.done)
|
|
for {
|
|
select {
|
|
case <-state.DoneCh:
|
|
return
|
|
|
|
case <-time.After(h.periodicUiTimer):
|
|
// Timer up, show status
|
|
}
|
|
|
|
var msg string
|
|
switch state.Op {
|
|
case uiResourceModify:
|
|
msg = "Still modifying..."
|
|
case uiResourceDestroy:
|
|
msg = "Still destroying..."
|
|
case uiResourceCreate:
|
|
msg = "Still creating..."
|
|
case uiResourceRead:
|
|
msg = "Still reading..."
|
|
case uiResourceUnknown:
|
|
return
|
|
}
|
|
|
|
idSuffix := ""
|
|
if state.IDKey != "" {
|
|
idSuffix = fmt.Sprintf("%s=%s, ", state.IDKey, truncateId(state.IDValue, maxIdLen))
|
|
}
|
|
|
|
h.println(fmt.Sprintf(
|
|
h.view.colorize.Color("[reset][bold]%s: %s [%s%s elapsed][reset]"),
|
|
state.DispAddr,
|
|
msg,
|
|
idSuffix,
|
|
time.Now().Round(time.Second).Sub(state.Start),
|
|
))
|
|
}
|
|
}
|
|
|
|
func (h *UiHook) PostApply(addr addrs.AbsResourceInstance, gen states.Generation, newState cty.Value, applyerr error) (terraform.HookAction, error) {
|
|
id := addr.String()
|
|
|
|
h.resourcesLock.Lock()
|
|
state := h.resources[id]
|
|
if state.DoneCh != nil {
|
|
close(state.DoneCh)
|
|
}
|
|
|
|
delete(h.resources, id)
|
|
h.resourcesLock.Unlock()
|
|
|
|
var stateIdSuffix string
|
|
if k, v := format.ObjectValueID(newState); k != "" && v != "" {
|
|
stateIdSuffix = fmt.Sprintf(" [%s=%s]", k, v)
|
|
}
|
|
|
|
var msg string
|
|
switch state.Op {
|
|
case uiResourceModify:
|
|
msg = "Modifications complete"
|
|
case uiResourceDestroy:
|
|
msg = "Destruction complete"
|
|
case uiResourceCreate:
|
|
msg = "Creation complete"
|
|
case uiResourceRead:
|
|
msg = "Read complete"
|
|
case uiResourceUnknown:
|
|
return terraform.HookActionContinue, nil
|
|
}
|
|
|
|
if applyerr != nil {
|
|
// Errors are collected and printed in ApplyCommand, no need to duplicate
|
|
return terraform.HookActionContinue, nil
|
|
}
|
|
|
|
addrStr := addr.String()
|
|
if depKey, ok := gen.(states.DeposedKey); ok {
|
|
addrStr = fmt.Sprintf("%s (deposed object %s)", addrStr, depKey)
|
|
}
|
|
|
|
colorized := fmt.Sprintf(
|
|
h.view.colorize.Color("[reset][bold]%s: %s after %s%s"),
|
|
addrStr, msg, time.Now().Round(time.Second).Sub(state.Start), stateIdSuffix)
|
|
|
|
h.println(colorized)
|
|
|
|
return terraform.HookActionContinue, nil
|
|
}
|
|
|
|
func (h *UiHook) PreProvisionInstanceStep(addr addrs.AbsResourceInstance, typeName string) (terraform.HookAction, error) {
|
|
h.println(fmt.Sprintf(
|
|
h.view.colorize.Color("[reset][bold]%s: Provisioning with '%s'...[reset]"),
|
|
addr, typeName,
|
|
))
|
|
return terraform.HookActionContinue, nil
|
|
}
|
|
|
|
func (h *UiHook) ProvisionOutput(addr addrs.AbsResourceInstance, typeName string, msg string) {
|
|
var buf bytes.Buffer
|
|
|
|
prefix := fmt.Sprintf(
|
|
h.view.colorize.Color("[reset][bold]%s (%s):[reset] "),
|
|
addr, typeName,
|
|
)
|
|
s := bufio.NewScanner(strings.NewReader(msg))
|
|
s.Split(scanLines)
|
|
for s.Scan() {
|
|
line := strings.TrimRightFunc(s.Text(), unicode.IsSpace)
|
|
if line != "" {
|
|
buf.WriteString(fmt.Sprintf("%s%s\n", prefix, line))
|
|
}
|
|
}
|
|
|
|
h.println(strings.TrimSpace(buf.String()))
|
|
}
|
|
|
|
func (h *UiHook) PreRefresh(addr addrs.AbsResourceInstance, gen states.Generation, priorState cty.Value) (terraform.HookAction, error) {
|
|
var stateIdSuffix string
|
|
if k, v := format.ObjectValueID(priorState); k != "" && v != "" {
|
|
stateIdSuffix = fmt.Sprintf(" [%s=%s]", k, v)
|
|
}
|
|
|
|
addrStr := addr.String()
|
|
if depKey, ok := gen.(states.DeposedKey); ok {
|
|
addrStr = fmt.Sprintf("%s (deposed object %s)", addrStr, depKey)
|
|
}
|
|
|
|
h.println(fmt.Sprintf(
|
|
h.view.colorize.Color("[reset][bold]%s: Refreshing state...%s"),
|
|
addrStr, stateIdSuffix))
|
|
return terraform.HookActionContinue, nil
|
|
}
|
|
|
|
func (h *UiHook) PreImportState(addr addrs.AbsResourceInstance, importID string) (terraform.HookAction, error) {
|
|
h.println(fmt.Sprintf(
|
|
h.view.colorize.Color("[reset][bold]%s: Importing from ID %q..."),
|
|
addr, importID,
|
|
))
|
|
return terraform.HookActionContinue, nil
|
|
}
|
|
|
|
func (h *UiHook) PostImportState(addr addrs.AbsResourceInstance, imported []providers.ImportedResource) (terraform.HookAction, error) {
|
|
h.println(fmt.Sprintf(
|
|
h.view.colorize.Color("[reset][bold][green]%s: Import prepared!"),
|
|
addr,
|
|
))
|
|
for _, s := range imported {
|
|
h.println(fmt.Sprintf(
|
|
h.view.colorize.Color("[reset][green] Prepared %s for import"),
|
|
s.TypeName,
|
|
))
|
|
}
|
|
|
|
return terraform.HookActionContinue, nil
|
|
}
|
|
|
|
// Wrap calls to the view so that concurrent calls do not interleave println.
|
|
func (h *UiHook) println(s string) {
|
|
h.viewLock.Lock()
|
|
defer h.viewLock.Unlock()
|
|
h.view.streams.Println(s)
|
|
}
|
|
|
|
// scanLines is basically copied from the Go standard library except
|
|
// we've modified it to also fine `\r`.
|
|
func scanLines(data []byte, atEOF bool) (advance int, token []byte, err error) {
|
|
if atEOF && len(data) == 0 {
|
|
return 0, nil, nil
|
|
}
|
|
if i := bytes.IndexByte(data, '\n'); i >= 0 {
|
|
// We have a full newline-terminated line.
|
|
return i + 1, dropCR(data[0:i]), nil
|
|
}
|
|
if i := bytes.IndexByte(data, '\r'); i >= 0 {
|
|
// We have a full carriage-return-terminated line.
|
|
return i + 1, dropCR(data[0:i]), nil
|
|
}
|
|
// If we're at EOF, we have a final, non-terminated line. Return it.
|
|
if atEOF {
|
|
return len(data), dropCR(data), nil
|
|
}
|
|
// Request more data.
|
|
return 0, nil, nil
|
|
}
|
|
|
|
// dropCR drops a terminal \r from the data.
|
|
func dropCR(data []byte) []byte {
|
|
if len(data) > 0 && data[len(data)-1] == '\r' {
|
|
return data[0 : len(data)-1]
|
|
}
|
|
return data
|
|
}
|
|
|
|
func truncateId(id string, maxLen int) string {
|
|
// Note that the id may contain multibyte characters.
|
|
// We need to truncate it to maxLen characters, not maxLen bytes.
|
|
rid := []rune(id)
|
|
totalLength := len(rid)
|
|
if totalLength <= maxLen {
|
|
return id
|
|
}
|
|
if maxLen < 5 {
|
|
// We don't shorten to less than 5 chars
|
|
// as that would be pointless with ... (3 chars)
|
|
maxLen = 5
|
|
}
|
|
|
|
dots := []rune("...")
|
|
partLen := maxLen / 2
|
|
|
|
leftIdx := partLen - 1
|
|
leftPart := rid[0:leftIdx]
|
|
|
|
rightIdx := totalLength - partLen - 1
|
|
|
|
overlap := maxLen - (partLen*2 + len(dots))
|
|
if overlap < 0 {
|
|
rightIdx -= overlap
|
|
}
|
|
|
|
rightPart := rid[rightIdx:]
|
|
|
|
return string(leftPart) + string(dots) + string(rightPart)
|
|
}
|