2014-07-01 19:02:13 +02:00
|
|
|
package command
|
|
|
|
|
|
|
|
import (
|
2014-07-12 05:38:03 +02:00
|
|
|
"bytes"
|
2014-07-01 19:02:13 +02:00
|
|
|
"flag"
|
|
|
|
"fmt"
|
2014-07-12 05:41:47 +02:00
|
|
|
"os"
|
2014-07-01 19:02:13 +02:00
|
|
|
"strings"
|
|
|
|
|
|
|
|
"github.com/hashicorp/terraform/config"
|
|
|
|
"github.com/hashicorp/terraform/digraph"
|
|
|
|
"github.com/hashicorp/terraform/terraform"
|
|
|
|
"github.com/mitchellh/cli"
|
|
|
|
)
|
|
|
|
|
|
|
|
// GraphCommand is a Command implementation that takes a Terraform
|
|
|
|
// configuration and outputs the dependency tree in graphical form.
|
|
|
|
type GraphCommand struct {
|
2014-07-03 20:46:40 +02:00
|
|
|
ContextOpts *terraform.ContextOpts
|
|
|
|
Ui cli.Ui
|
2014-07-01 19:02:13 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (c *GraphCommand) Run(args []string) int {
|
|
|
|
cmdFlags := flag.NewFlagSet("graph", flag.ContinueOnError)
|
|
|
|
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-07-12 05:38:03 +02:00
|
|
|
conf, err := config.LoadDir(path)
|
2014-07-01 19:02:13 +02:00
|
|
|
if err != nil {
|
|
|
|
c.Ui.Error(fmt.Sprintf("Error loading config: %s", err))
|
|
|
|
return 1
|
|
|
|
}
|
|
|
|
|
|
|
|
g, err := terraform.Graph(&terraform.GraphOpts{
|
|
|
|
Config: conf,
|
2014-07-03 20:46:40 +02:00
|
|
|
Providers: c.ContextOpts.Providers,
|
2014-07-01 19:02:13 +02:00
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
c.Ui.Error(fmt.Sprintf("Error creating graph: %s", err))
|
|
|
|
return 1
|
|
|
|
}
|
|
|
|
|
2014-07-12 05:38:03 +02:00
|
|
|
buf := new(bytes.Buffer)
|
2014-07-01 19:02:13 +02:00
|
|
|
nodes := make([]digraph.Node, len(g.Nouns))
|
|
|
|
for i, n := range g.Nouns {
|
|
|
|
nodes[i] = n
|
|
|
|
}
|
2014-07-12 05:38:03 +02:00
|
|
|
digraph.GenerateDot(nodes, buf)
|
|
|
|
|
|
|
|
c.Ui.Output(buf.String())
|
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-07-01 19:02:13 +02:00
|
|
|
`
|
|
|
|
return strings.TrimSpace(helpText)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *GraphCommand) Synopsis() string {
|
|
|
|
return "Output visual graph of Terraform resources"
|
|
|
|
}
|