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"
|
2015-02-20 07:58:42 +01:00
|
|
|
|
2016-11-09 15:58:52 +01:00
|
|
|
"github.com/hashicorp/terraform/dag"
|
2015-02-20 07:58:42 +01: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
|
2015-04-23 20:24:26 +02:00
|
|
|
var verbose bool
|
|
|
|
var drawCycles bool
|
2016-12-04 00:00:34 +01:00
|
|
|
var graphTypeStr string
|
2014-09-25 02:36:27 +02:00
|
|
|
|
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)
|
2015-04-30 21:05:39 +02:00
|
|
|
c.addModuleDepthFlag(cmdFlags, &moduleDepth)
|
2015-04-23 20:24:26 +02:00
|
|
|
cmdFlags.BoolVar(&verbose, "verbose", false, "verbose")
|
|
|
|
cmdFlags.BoolVar(&drawCycles, "draw-cycles", false, "draw-cycles")
|
2016-12-04 00:00:34 +01:00
|
|
|
cmdFlags.StringVar(&graphTypeStr, "type", "", "type")
|
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
|
|
|
}
|
|
|
|
|
2016-12-04 00:00:34 +01:00
|
|
|
ctx, planFile, err := c.Context(contextOpts{
|
2014-09-22 19:56:50 +02:00
|
|
|
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
|
|
|
|
}
|
|
|
|
|
2016-12-04 00:00:34 +01:00
|
|
|
// Determine the graph type
|
|
|
|
graphType := terraform.GraphTypePlan
|
|
|
|
if planFile {
|
|
|
|
graphType = terraform.GraphTypeApply
|
|
|
|
}
|
|
|
|
|
|
|
|
if graphTypeStr != "" {
|
|
|
|
v, ok := terraform.GraphTypeMap[graphTypeStr]
|
|
|
|
if !ok {
|
|
|
|
c.Ui.Error(fmt.Sprintf("Invalid graph type requested: %s", graphTypeStr))
|
|
|
|
return 1
|
|
|
|
}
|
|
|
|
|
|
|
|
graphType = v
|
|
|
|
}
|
|
|
|
|
2015-04-23 20:24:26 +02:00
|
|
|
// Skip validation during graph generation - we want to see the graph even if
|
|
|
|
// it is invalid for some reason.
|
2016-12-04 00:00:34 +01:00
|
|
|
g, err := ctx.Graph(graphType, &terraform.ContextGraphOpts{
|
2015-04-23 20:24:26 +02:00
|
|
|
Verbose: verbose,
|
|
|
|
Validate: false,
|
2015-04-23 17:52:31 +02:00
|
|
|
})
|
2014-07-01 19:02:13 +02:00
|
|
|
if err != nil {
|
|
|
|
c.Ui.Error(fmt.Sprintf("Error creating graph: %s", err))
|
|
|
|
return 1
|
|
|
|
}
|
|
|
|
|
2016-11-09 15:58:52 +01:00
|
|
|
graphStr, err := terraform.GraphDot(g, &dag.DotOpts{
|
2015-04-23 20:24:26 +02:00
|
|
|
DrawCycles: drawCycles,
|
|
|
|
MaxDepth: moduleDepth,
|
|
|
|
Verbose: verbose,
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
c.Ui.Error(fmt.Sprintf("Error converting graph: %s", err))
|
|
|
|
return 1
|
|
|
|
}
|
|
|
|
|
|
|
|
c.Ui.Output(graphStr)
|
2014-07-01 19:02:13 +02:00
|
|
|
|
|
|
|
return 0
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *GraphCommand) Help() string {
|
|
|
|
helpText := `
|
2015-04-23 20:24:26 +02:00
|
|
|
Usage: terraform graph [options] [DIR]
|
2014-07-01 19:02:13 +02:00
|
|
|
|
2016-12-04 00:00:34 +01:00
|
|
|
Outputs the visual execution graph of Terraform resources according to
|
2015-04-23 20:24:26 +02:00
|
|
|
configuration files in DIR (or the current directory if omitted).
|
2014-07-01 19:02:13 +02:00
|
|
|
|
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.
|
|
|
|
|
2016-12-04 00:00:34 +01:00
|
|
|
The -type flag can be used to control the type of graph shown. Terraform
|
|
|
|
creates different graphs for different operations. See the options below
|
|
|
|
for the list of types supported. The default type is "plan" if a
|
|
|
|
configuration is given, and "apply" if a plan file is passed as an
|
|
|
|
argument.
|
2014-09-25 02:36:27 +02:00
|
|
|
|
2016-12-04 00:00:34 +01:00
|
|
|
Options:
|
2015-04-23 20:24:26 +02:00
|
|
|
|
2016-12-04 00:00:34 +01:00
|
|
|
-draw-cycles Highlight any cycles in the graph with colored edges.
|
|
|
|
This helps when diagnosing cycle errors.
|
2014-09-25 02:36:27 +02:00
|
|
|
|
2016-12-04 00:00:34 +01:00
|
|
|
-no-color If specified, output won't contain any color.
|
2015-06-22 14:14:01 +02:00
|
|
|
|
2016-12-04 00:00:34 +01:00
|
|
|
-type=plan Type of graph to output. Can be: plan, plan-destroy, apply,
|
|
|
|
legacy.
|
2015-06-22 14:14:01 +02:00
|
|
|
|
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
|
|
|
}
|