blob: fcc2883065e536d269cf6c2d8c29459fc4e78505 [file] [log] [blame]
// Copyright (c) HashiCorp, Inc.
// SPDX-License-Identifier: BUSL-1.1
package stackmigrate
import (
"fmt"
"os"
"path/filepath"
"github.com/zclconf/go-cty/cty"
"github.com/hashicorp/terraform-svchost/disco"
"github.com/hashicorp/terraform/internal/backend"
backendInit "github.com/hashicorp/terraform/internal/backend/init"
"github.com/hashicorp/terraform/internal/backend/local"
"github.com/hashicorp/terraform/internal/backend/remote"
"github.com/hashicorp/terraform/internal/command"
"github.com/hashicorp/terraform/internal/command/clistate"
"github.com/hashicorp/terraform/internal/command/workdir"
"github.com/hashicorp/terraform/internal/states"
"github.com/hashicorp/terraform/internal/states/statemgr"
"github.com/hashicorp/terraform/internal/tfdiags"
)
type Loader struct {
Discovery *disco.Disco
}
var (
WorkspaceNameEnvVar = "TF_WORKSPACE"
)
// LoadState loads a state from the given configPath. The configuration at configPath
// must have been initialized via `terraform init` before calling this function.
// The function returns an empty state even if there are errors.
func (l *Loader) LoadState(configPath string) (*states.State, tfdiags.Diagnostics) {
var diags tfdiags.Diagnostics
state := states.NewState()
workingDirectory, workspace, err := l.loadWorkingDir(configPath)
if err != nil {
return state, diags.Append(fmt.Errorf("error loading working directory: %s", err))
}
backendInit.Init(l.Discovery)
// First, we'll load the "backend state". This should have been initialised
// by the `terraform init` command, and contains the configuration for the
// backend that we're using.
var backendState *workdir.BackendStateFile
backendStatePath := filepath.Join(workingDirectory.DataDir(), ".terraform.tfstate")
st := &clistate.LocalState{Path: backendStatePath}
// If the backend state file is not provided, RefreshState will
// return nil error and State will be empty.
// In this case, we assume that we're using a local backend.
if err := st.RefreshState(); err != nil {
diags = diags.Append(fmt.Errorf("error loading backend state: %s", err))
return state, diags
}
backendState = st.State()
// Now that we have the backend state, we can initialise the backend itself
// based on what we had from the `terraform init` command.
var backend backend.Backend
var backendConfig cty.Value
// the absence of backend state file indicates a local backend
if backendState == nil {
backend = local.New()
backendConfig = cty.ObjectVal(map[string]cty.Value{
"path": cty.StringVal(fmt.Sprintf("%s/%s", configPath, "terraform.tfstate")),
"workspace_dir": cty.StringVal(configPath),
})
} else {
initFn := backendInit.Backend(backendState.Backend.Type)
if initFn == nil {
diags = diags.Append(fmt.Errorf("unknown backend type %q", backendState.Backend.Type))
return state, diags
}
backend = initFn()
schema := backend.ConfigSchema()
config, err := backendState.Backend.Config(schema)
if err != nil {
diags = diags.Append(tfdiags.Sourceless(
tfdiags.Error,
"Failed to decode current backend config",
fmt.Sprintf("The backend configuration created by the most recent run of \"terraform init\" could not be decoded: %s. The configuration may have been initialized by an earlier version that used an incompatible configuration structure. Run \"terraform init -reconfigure\" to force re-initialization of the backend.", err),
))
return state, diags
}
var moreDiags tfdiags.Diagnostics
backendConfig, moreDiags = backend.PrepareConfig(config)
diags = diags.Append(moreDiags)
if moreDiags.HasErrors() {
return state, diags
}
// it's safe to ignore terraform version conflict between the local and remote environments,
// as we are only reading the state
if backendR, ok := backend.(*remote.Remote); ok {
backendR.IgnoreVersionConflict()
}
}
// Now that we have the backend and its configuration, we can configure it.
moreDiags := backend.Configure(backendConfig)
diags = diags.Append(moreDiags)
if moreDiags.HasErrors() {
return state, diags
}
// The backend is initialised and configured, so now we can load the state
// from the backend.
stateManager, err := backend.StateMgr(workspace)
if err != nil {
diags = diags.Append(fmt.Errorf("error loading state: %s", err))
return state, diags
}
// We'll lock the backend here to ensure that we don't have any concurrent
// operations on the state. If this fails, we'll return an error and the
// user should retry the migration later when nothing is currently updating
// the state.
id, err := stateManager.Lock(statemgr.NewLockInfo())
if err != nil {
diags = diags.Append(tfdiags.Sourceless(tfdiags.Error, "Failed to lock state", fmt.Sprintf("The state is currently locked by another operation: %s. Please retry the migration later.", err)))
return state, diags
}
// Remember to unlock the state when we're done.
defer func() {
// Remember to unlock the state when we're done.
if err := stateManager.Unlock(id); err != nil {
// If we couldn't unlock the state, we'll warn about that but the
// migration can actually continue.
diags = diags.Append(tfdiags.Sourceless(tfdiags.Warning, "Failed to unlock state", fmt.Sprintf("The state was successfully loaded but could not be unlocked: %s. The migration can continue but the state many need to be unlocked manually.", err)))
}
}()
if err := stateManager.RefreshState(); err != nil {
diags = diags.Append(fmt.Errorf("error loading state: %s", err))
return state, diags
}
state = stateManager.State()
return state, diags
}
func (l *Loader) loadWorkingDir(configPath string) (*workdir.Dir, string, error) {
// load the state specified by this configuration
workingDirectory := workdir.NewDir(configPath)
if data := os.Getenv("TF_DATA_DIR"); len(data) > 0 {
workingDirectory.OverrideDataDir(data)
}
meta := &command.Meta{WorkingDir: workingDirectory}
// Load the currently active workspace from the environment, defaulting
// to the default workspace if not set.
workspace, err := meta.Workspace()
if err != nil {
return nil, "", fmt.Errorf("failed to load workspace: %s", err)
}
return workingDirectory, workspace, nil
}