mirror of
https://github.com/hashicorp/terraform.git
synced 2026-02-19 02:39:17 -05:00
Some checks are pending
build / Determine intended Terraform version (push) Waiting to run
build / Determine Go toolchain version (push) Waiting to run
build / Generate release metadata (push) Blocked by required conditions
build / Build for freebsd_386 (push) Blocked by required conditions
build / Build for linux_386 (push) Blocked by required conditions
build / Build for openbsd_386 (push) Blocked by required conditions
build / Build for windows_386 (push) Blocked by required conditions
build / Build for darwin_amd64 (push) Blocked by required conditions
build / Build for freebsd_amd64 (push) Blocked by required conditions
build / Build for linux_amd64 (push) Blocked by required conditions
build / Build for openbsd_amd64 (push) Blocked by required conditions
build / Build for solaris_amd64 (push) Blocked by required conditions
build / Build for windows_amd64 (push) Blocked by required conditions
build / Build for freebsd_arm (push) Blocked by required conditions
build / Build for linux_arm (push) Blocked by required conditions
build / Build for darwin_arm64 (push) Blocked by required conditions
build / Build for linux_arm64 (push) Blocked by required conditions
build / Build for windows_arm64 (push) Blocked by required conditions
build / Build Docker image for linux_386 (push) Blocked by required conditions
build / Build Docker image for linux_amd64 (push) Blocked by required conditions
build / Build Docker image for linux_arm (push) Blocked by required conditions
build / Build Docker image for linux_arm64 (push) Blocked by required conditions
build / Build e2etest for linux_386 (push) Blocked by required conditions
build / Build e2etest for windows_386 (push) Blocked by required conditions
build / Build e2etest for darwin_amd64 (push) Blocked by required conditions
build / Build e2etest for linux_amd64 (push) Blocked by required conditions
build / Build e2etest for windows_amd64 (push) Blocked by required conditions
build / Build e2etest for linux_arm (push) Blocked by required conditions
build / Build e2etest for darwin_arm64 (push) Blocked by required conditions
build / Build e2etest for linux_arm64 (push) Blocked by required conditions
build / Run e2e test for linux_386 (push) Blocked by required conditions
build / Run e2e test for windows_386 (push) Blocked by required conditions
build / Run e2e test for darwin_amd64 (push) Blocked by required conditions
build / Run e2e test for linux_amd64 (push) Blocked by required conditions
build / Run e2e test for windows_amd64 (push) Blocked by required conditions
build / Run e2e test for linux_arm (push) Blocked by required conditions
build / Run e2e test for linux_arm64 (push) Blocked by required conditions
build / Run terraform-exec test for linux amd64 (push) Blocked by required conditions
Quick Checks / Unit Tests (push) Waiting to run
Quick Checks / Race Tests (push) Waiting to run
Quick Checks / End-to-end Tests (push) Waiting to run
Quick Checks / Code Consistency Checks (push) Waiting to run
Prior to this change, when running a `state show` command with an address for a resource that existed in state but didn't match the current schema of the resource, users would see the "Failed to marshal" error but would also see output saying "The state file is empty. No resources are represented". This is misleading, as the state file isn't empty. Also, the command would exit with code 0 despite the error.
207 lines
5.6 KiB
Go
207 lines
5.6 KiB
Go
// Copyright (c) HashiCorp, Inc.
|
|
// SPDX-License-Identifier: BUSL-1.1
|
|
|
|
package command
|
|
|
|
import (
|
|
"fmt"
|
|
"os"
|
|
"strings"
|
|
|
|
"github.com/hashicorp/cli"
|
|
|
|
"github.com/hashicorp/terraform/internal/addrs"
|
|
"github.com/hashicorp/terraform/internal/backend/backendrun"
|
|
"github.com/hashicorp/terraform/internal/command/arguments"
|
|
"github.com/hashicorp/terraform/internal/command/jsonformat"
|
|
"github.com/hashicorp/terraform/internal/command/jsonprovider"
|
|
"github.com/hashicorp/terraform/internal/command/jsonstate"
|
|
"github.com/hashicorp/terraform/internal/states"
|
|
"github.com/hashicorp/terraform/internal/states/statefile"
|
|
)
|
|
|
|
// StateShowCommand is a Command implementation that shows a single resource.
|
|
type StateShowCommand struct {
|
|
Meta
|
|
StateMeta
|
|
}
|
|
|
|
func (c *StateShowCommand) Run(args []string) int {
|
|
args = c.Meta.process(args)
|
|
cmdFlags := c.Meta.defaultFlagSet("state show")
|
|
cmdFlags.StringVar(&c.Meta.statePath, "state", "", "path")
|
|
if err := cmdFlags.Parse(args); err != nil {
|
|
c.Streams.Eprintf("Error parsing command-line flags: %s\n", err.Error())
|
|
return 1
|
|
}
|
|
args = cmdFlags.Args()
|
|
if len(args) != 1 {
|
|
c.Streams.Eprint("Exactly one argument expected.\n")
|
|
return cli.RunResultHelp
|
|
}
|
|
|
|
// Check for user-supplied plugin path
|
|
var err error
|
|
if c.pluginPath, err = c.loadPluginPath(); err != nil {
|
|
c.Streams.Eprintf("Error loading plugin path: %\n", err)
|
|
return 1
|
|
}
|
|
|
|
// Load the backend
|
|
view := arguments.ViewHuman
|
|
b, diags := c.backend(".", view)
|
|
if diags.HasErrors() {
|
|
c.showDiagnostics(diags)
|
|
return 1
|
|
}
|
|
|
|
// We require a local backend
|
|
local, ok := b.(backendrun.Local)
|
|
if !ok {
|
|
c.Streams.Eprint(ErrUnsupportedLocalOp)
|
|
return 1
|
|
}
|
|
|
|
// This is a read-only command
|
|
c.ignoreRemoteVersionConflict(b)
|
|
|
|
// Check if the address can be parsed
|
|
addr, addrDiags := addrs.ParseAbsResourceInstanceStr(args[0])
|
|
if addrDiags.HasErrors() {
|
|
c.Streams.Eprintln(fmt.Sprintf(errParsingAddress, args[0]))
|
|
return 1
|
|
}
|
|
|
|
// We expect the config dir to always be the cwd
|
|
cwd, err := os.Getwd()
|
|
if err != nil {
|
|
c.Streams.Eprintf("Error getting cwd: %s\n", err)
|
|
return 1
|
|
}
|
|
|
|
// Build the operation (required to get the schemas)
|
|
opReq := c.Operation(b, arguments.ViewHuman)
|
|
opReq.AllowUnsetVariables = true
|
|
opReq.ConfigDir = cwd
|
|
|
|
opReq.ConfigLoader, err = c.initConfigLoader()
|
|
if err != nil {
|
|
c.Streams.Eprintf("Error initializing config loader: %s\n", err)
|
|
return 1
|
|
}
|
|
|
|
// Get the context (required to get the schemas)
|
|
lr, _, ctxDiags := local.LocalRun(opReq)
|
|
if ctxDiags.HasErrors() {
|
|
c.View.Diagnostics(ctxDiags)
|
|
return 1
|
|
}
|
|
|
|
// Get the schemas from the context
|
|
schemas, diags := lr.Core.Schemas(lr.Config, lr.InputState)
|
|
if diags.HasErrors() {
|
|
c.View.Diagnostics(diags)
|
|
return 1
|
|
}
|
|
|
|
// Get the state
|
|
env, err := c.Workspace()
|
|
if err != nil {
|
|
c.Streams.Eprintf("Error selecting workspace: %s\n", err)
|
|
return 1
|
|
}
|
|
stateMgr, sDiags := b.StateMgr(env)
|
|
if sDiags.HasErrors() {
|
|
c.Streams.Eprintln(fmt.Sprintf(errStateLoadingState, sDiags.Err()))
|
|
return 1
|
|
}
|
|
if err := stateMgr.RefreshState(); err != nil {
|
|
c.Streams.Eprintf("Failed to refresh state: %s\n", err)
|
|
return 1
|
|
}
|
|
|
|
state := stateMgr.State()
|
|
if state == nil {
|
|
c.Streams.Eprintln(errStateNotFound)
|
|
return 1
|
|
}
|
|
|
|
is := state.ResourceInstance(addr)
|
|
if !is.HasCurrent() {
|
|
c.Streams.Eprintln(errNoInstanceFound)
|
|
return 1
|
|
}
|
|
|
|
// check if the resource has a configured provider, otherwise this will use the default provider
|
|
rs := state.Resource(addr.ContainingResource())
|
|
absPc := addrs.AbsProviderConfig{
|
|
Provider: rs.ProviderConfig.Provider,
|
|
Alias: rs.ProviderConfig.Alias,
|
|
Module: addrs.RootModule,
|
|
}
|
|
singleInstance := states.NewState()
|
|
singleInstance.EnsureModule(addr.Module).SetResourceInstanceCurrent(
|
|
addr.Resource,
|
|
is.Current,
|
|
absPc,
|
|
)
|
|
|
|
root, outputs, err := jsonstate.MarshalForRenderer(statefile.New(singleInstance, "", 0), schemas)
|
|
if err != nil {
|
|
c.Streams.Eprintf("Failed to marshal state to json: %s", err)
|
|
return 1
|
|
}
|
|
|
|
jstate := jsonformat.State{
|
|
StateFormatVersion: jsonstate.FormatVersion,
|
|
ProviderFormatVersion: jsonprovider.FormatVersion,
|
|
RootModule: root,
|
|
RootModuleOutputs: outputs,
|
|
ProviderSchemas: jsonprovider.MarshalForRenderer(schemas),
|
|
}
|
|
|
|
renderer := jsonformat.Renderer{
|
|
Streams: c.Streams,
|
|
Colorize: c.Colorize(),
|
|
RunningInAutomation: c.RunningInAutomation,
|
|
}
|
|
|
|
renderer.RenderHumanState(jstate)
|
|
return 0
|
|
}
|
|
|
|
func (c *StateShowCommand) Help() string {
|
|
helpText := `
|
|
Usage: terraform [global options] state show [options] ADDRESS
|
|
|
|
Shows the attributes of a resource in the Terraform state.
|
|
|
|
This command shows the attributes of a single resource in the Terraform
|
|
state. The address argument must be used to specify a single resource.
|
|
You can view the list of available resources with "terraform state list".
|
|
|
|
Options:
|
|
|
|
-state=statefile Path to a Terraform state file to use to look
|
|
up Terraform-managed resources. By default it will
|
|
use the state "terraform.tfstate" if it exists.
|
|
|
|
`
|
|
return strings.TrimSpace(helpText)
|
|
}
|
|
|
|
func (c *StateShowCommand) Synopsis() string {
|
|
return "Show a resource in the state"
|
|
}
|
|
|
|
const errNoInstanceFound = `No instance found for the given address!
|
|
|
|
This command requires that the address references one specific instance.
|
|
To view the available instances, use "terraform state list". Please modify
|
|
the address to reference a specific instance.`
|
|
|
|
const errParsingAddress = `Error parsing instance address: %s
|
|
|
|
This command requires that the address references one specific instance.
|
|
To view the available instances, use "terraform state list". Please modify
|
|
the address to reference a specific instance.`
|