2016-03-22 18:41:02 +01:00
|
|
|
package command
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"strings"
|
|
|
|
|
2019-03-16 03:51:26 +01:00
|
|
|
"github.com/hashicorp/terraform/addrs"
|
2018-10-17 23:01:15 +02:00
|
|
|
"github.com/hashicorp/terraform/states"
|
2019-03-16 03:51:26 +01:00
|
|
|
"github.com/hashicorp/terraform/tfdiags"
|
2016-03-22 18:41:02 +01:00
|
|
|
"github.com/mitchellh/cli"
|
|
|
|
)
|
|
|
|
|
|
|
|
// StateListCommand is a Command implementation that lists the resources
|
|
|
|
// within a state file.
|
|
|
|
type StateListCommand struct {
|
2017-03-01 16:10:47 +01:00
|
|
|
Meta
|
2017-02-28 19:13:03 +01:00
|
|
|
StateMeta
|
2016-03-22 18:41:02 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
func (c *StateListCommand) Run(args []string) int {
|
2020-04-01 21:01:08 +02:00
|
|
|
args = c.Meta.process(args)
|
2019-04-15 18:22:07 +02:00
|
|
|
var statePath string
|
2018-11-21 15:35:27 +01:00
|
|
|
cmdFlags := c.Meta.defaultFlagSet("state list")
|
2019-04-15 18:22:07 +02:00
|
|
|
cmdFlags.StringVar(&statePath, "state", "", "path")
|
2018-10-17 23:01:15 +02:00
|
|
|
lookupId := cmdFlags.String("id", "", "Restrict output to paths with a resource having the specified ID.")
|
2016-03-22 18:41:02 +01:00
|
|
|
if err := cmdFlags.Parse(args); err != nil {
|
2019-08-16 14:31:21 +02:00
|
|
|
c.Ui.Error(fmt.Sprintf("Error parsing command-line flags: %s\n", err.Error()))
|
2016-03-22 18:41:02 +01:00
|
|
|
return cli.RunResultHelp
|
|
|
|
}
|
|
|
|
args = cmdFlags.Args()
|
|
|
|
|
2019-04-15 18:22:07 +02:00
|
|
|
if statePath != "" {
|
|
|
|
c.Meta.statePath = statePath
|
|
|
|
}
|
|
|
|
|
2017-01-19 05:50:45 +01:00
|
|
|
// Load the backend
|
2018-03-28 00:31:05 +02:00
|
|
|
b, backendDiags := c.Backend(nil)
|
|
|
|
if backendDiags.HasErrors() {
|
|
|
|
c.showDiagnostics(backendDiags)
|
2017-01-19 05:50:45 +01:00
|
|
|
return 1
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get the state
|
2020-06-16 18:23:15 +02:00
|
|
|
env, err := c.Workspace()
|
|
|
|
if err != nil {
|
|
|
|
c.Ui.Error(fmt.Sprintf("Error selecting workspace: %s", err))
|
|
|
|
return 1
|
|
|
|
}
|
2018-10-17 23:01:15 +02:00
|
|
|
stateMgr, err := b.StateMgr(env)
|
2017-01-19 05:50:45 +01:00
|
|
|
if err != nil {
|
2018-10-22 15:52:53 +02:00
|
|
|
c.Ui.Error(fmt.Sprintf(errStateLoadingState, err))
|
2017-01-19 05:50:45 +01:00
|
|
|
return 1
|
2016-03-22 18:41:02 +01:00
|
|
|
}
|
2018-10-17 23:01:15 +02:00
|
|
|
if err := stateMgr.RefreshState(); err != nil {
|
2019-04-15 18:22:07 +02:00
|
|
|
c.Ui.Error(fmt.Sprintf("Failed to load state: %s", err))
|
2017-02-22 05:35:43 +01:00
|
|
|
return 1
|
|
|
|
}
|
|
|
|
|
2018-10-17 23:01:15 +02:00
|
|
|
state := stateMgr.State()
|
|
|
|
if state == nil {
|
2016-03-22 18:41:02 +01:00
|
|
|
c.Ui.Error(fmt.Sprintf(errStateNotFound))
|
|
|
|
return 1
|
|
|
|
}
|
|
|
|
|
2019-03-16 03:51:26 +01:00
|
|
|
var addrs []addrs.AbsResourceInstance
|
|
|
|
var diags tfdiags.Diagnostics
|
|
|
|
if len(args) == 0 {
|
|
|
|
addrs, diags = c.lookupAllResourceInstanceAddrs(state)
|
|
|
|
} else {
|
|
|
|
addrs, diags = c.lookupResourceInstanceAddrs(state, args...)
|
|
|
|
}
|
|
|
|
if diags.HasErrors() {
|
|
|
|
c.showDiagnostics(diags)
|
|
|
|
return 1
|
2016-03-22 18:41:02 +01:00
|
|
|
}
|
|
|
|
|
2019-03-16 03:51:26 +01:00
|
|
|
for _, addr := range addrs {
|
|
|
|
if is := state.ResourceInstance(addr); is != nil {
|
2018-10-17 23:01:15 +02:00
|
|
|
if *lookupId == "" || *lookupId == states.LegacyInstanceObjectID(is.Current) {
|
2019-03-16 03:51:26 +01:00
|
|
|
c.Ui.Output(addr.String())
|
2018-02-26 19:54:48 +01:00
|
|
|
}
|
2016-03-22 18:41:02 +01:00
|
|
|
}
|
2018-10-17 23:01:15 +02:00
|
|
|
}
|
2016-03-22 18:41:02 +01:00
|
|
|
|
2019-03-16 03:51:26 +01:00
|
|
|
c.showDiagnostics(diags)
|
|
|
|
|
2016-03-22 18:41:02 +01:00
|
|
|
return 0
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *StateListCommand) Help() string {
|
|
|
|
helpText := `
|
2019-03-16 03:51:26 +01:00
|
|
|
Usage: terraform state list [options] [address...]
|
2016-03-22 18:41:02 +01:00
|
|
|
|
|
|
|
List resources in the Terraform state.
|
|
|
|
|
2019-03-16 03:51:26 +01:00
|
|
|
This command lists resource instances in the Terraform state. The address
|
|
|
|
argument can be used to filter the instances by resource or module. If
|
|
|
|
no pattern is given, all resource instances are listed.
|
2016-03-22 18:41:02 +01:00
|
|
|
|
2019-03-16 03:51:26 +01:00
|
|
|
The addresses must either be module addresses or absolute resource
|
|
|
|
addresses, such as:
|
|
|
|
aws_instance.example
|
|
|
|
module.example
|
|
|
|
module.example.module.child
|
|
|
|
module.example.aws_instance.example
|
2016-03-22 18:41:02 +01:00
|
|
|
|
2019-03-16 03:51:26 +01:00
|
|
|
An error will be returned if any of the resources or modules given as
|
|
|
|
filter addresses do not exist in the state.
|
2016-03-22 18:41:02 +01:00
|
|
|
|
|
|
|
Options:
|
|
|
|
|
|
|
|
-state=statefile Path to a Terraform state file to use to look
|
2019-03-16 03:51:26 +01:00
|
|
|
up Terraform-managed resources. By default, Terraform
|
|
|
|
will consult the state of the currently-selected
|
|
|
|
workspace.
|
2016-03-22 18:41:02 +01:00
|
|
|
|
2019-03-16 03:51:26 +01:00
|
|
|
-id=ID Filters the results to include only instances whose
|
|
|
|
resource types have an attribute named "id" whose value
|
|
|
|
equals the given id string.
|
2018-02-26 19:54:48 +01:00
|
|
|
|
2016-03-22 18:41:02 +01:00
|
|
|
`
|
|
|
|
return strings.TrimSpace(helpText)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *StateListCommand) Synopsis() string {
|
|
|
|
return "List resources in the state"
|
|
|
|
}
|
|
|
|
|
|
|
|
const errStateLoadingState = `Error loading the state: %[1]s
|
|
|
|
|
|
|
|
Please ensure that your Terraform state exists and that you've
|
|
|
|
configured it properly. You can use the "-state" flag to point
|
|
|
|
Terraform at another state file.`
|
|
|
|
|
|
|
|
const errStateNotFound = `No state file was found!
|
|
|
|
|
|
|
|
State management commands require a state file. Run this command
|
|
|
|
in a directory where Terraform has been run or use the -state flag
|
|
|
|
to point the command to a specific state location.`
|