2014-07-01 19:02:13 +02:00
|
|
|
package command
|
|
|
|
|
|
|
|
import (
|
|
|
|
"flag"
|
|
|
|
"fmt"
|
2014-07-12 05:41:47 +02:00
|
|
|
"os"
|
2014-07-01 19:02:13 +02:00
|
|
|
"strings"
|
|
|
|
|
2014-07-14 20:34:52 +02:00
|
|
|
"github.com/hashicorp/terraform/terraform"
|
2014-07-01 19:02:13 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
// GraphCommand is a Command implementation that takes a Terraform
|
|
|
|
// configuration and outputs the dependency tree in graphical form.
|
|
|
|
type GraphCommand struct {
|
2014-07-13 05:37:30 +02:00
|
|
|
Meta
|
2014-07-01 19:02:13 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (c *GraphCommand) Run(args []string) int {
|
2014-09-25 02:36:27 +02:00
|
|
|
var moduleDepth int
|
|
|
|
|
2014-08-05 18:32:01 +02:00
|
|
|
args = c.Meta.process(args, false)
|
2014-07-13 05:37:30 +02:00
|
|
|
|
2014-07-01 19:02:13 +02:00
|
|
|
cmdFlags := flag.NewFlagSet("graph", flag.ContinueOnError)
|
2014-09-25 02:36:27 +02:00
|
|
|
cmdFlags.IntVar(&moduleDepth, "module-depth", 0, "module-depth")
|
2014-07-01 19:02:13 +02:00
|
|
|
cmdFlags.Usage = func() { c.Ui.Error(c.Help()) }
|
|
|
|
if err := cmdFlags.Parse(args); err != nil {
|
|
|
|
return 1
|
|
|
|
}
|
|
|
|
|
2014-07-12 05:41:47 +02:00
|
|
|
var path string
|
2014-07-01 19:02:13 +02:00
|
|
|
args = cmdFlags.Args()
|
2014-07-12 05:41:47 +02:00
|
|
|
if len(args) > 1 {
|
2014-07-01 19:02:13 +02:00
|
|
|
c.Ui.Error("The graph command expects one argument.\n")
|
|
|
|
cmdFlags.Usage()
|
|
|
|
return 1
|
2014-07-12 05:41:47 +02:00
|
|
|
} else if len(args) == 1 {
|
|
|
|
path = args[0]
|
|
|
|
} else {
|
|
|
|
var err error
|
|
|
|
path, err = os.Getwd()
|
|
|
|
if err != nil {
|
|
|
|
c.Ui.Error(fmt.Sprintf("Error getting pwd: %s", err))
|
|
|
|
}
|
2014-07-01 19:02:13 +02:00
|
|
|
}
|
|
|
|
|
2014-09-22 19:56:50 +02:00
|
|
|
ctx, _, err := c.Context(contextOpts{
|
|
|
|
Path: path,
|
|
|
|
StatePath: "",
|
|
|
|
})
|
2014-07-01 19:02:13 +02:00
|
|
|
if err != nil {
|
2014-07-13 04:25:50 +02:00
|
|
|
c.Ui.Error(fmt.Sprintf("Error loading Terraform: %s", err))
|
2014-07-01 19:02:13 +02:00
|
|
|
return 1
|
|
|
|
}
|
|
|
|
|
2014-07-13 04:25:50 +02:00
|
|
|
g, err := ctx.Graph()
|
2014-07-01 19:02:13 +02:00
|
|
|
if err != nil {
|
|
|
|
c.Ui.Error(fmt.Sprintf("Error creating graph: %s", err))
|
|
|
|
return 1
|
|
|
|
}
|
|
|
|
|
2014-09-25 02:36:27 +02:00
|
|
|
opts := &terraform.GraphDotOpts{
|
|
|
|
ModuleDepth: moduleDepth,
|
|
|
|
}
|
|
|
|
|
|
|
|
c.Ui.Output(terraform.GraphDot(g, opts))
|
2014-07-01 19:02:13 +02:00
|
|
|
|
|
|
|
return 0
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *GraphCommand) Help() string {
|
|
|
|
helpText := `
|
|
|
|
Usage: terraform graph [options] PATH
|
|
|
|
|
|
|
|
Outputs the visual graph of Terraform resources. If the path given is
|
|
|
|
the path to a configuration, the dependency graph of the resources are
|
|
|
|
shown. If the path is a plan file, then the dependency graph of the
|
|
|
|
plan itself is shown.
|
|
|
|
|
2014-07-12 05:38:03 +02:00
|
|
|
The graph is outputted in DOT format. The typical program that can
|
|
|
|
read this format is GraphViz, but many web services are also available
|
|
|
|
to read this format.
|
|
|
|
|
2014-09-25 02:36:27 +02:00
|
|
|
Options:
|
|
|
|
|
|
|
|
-module-depth=n The maximum depth to expand modules. By default this is
|
|
|
|
zero, which will not expand modules at all.
|
|
|
|
|
2014-07-01 19:02:13 +02:00
|
|
|
`
|
|
|
|
return strings.TrimSpace(helpText)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *GraphCommand) Synopsis() string {
|
2014-07-13 04:28:38 +02:00
|
|
|
return "Create a visual graph of Terraform resources"
|
2014-07-01 19:02:13 +02:00
|
|
|
}
|