2014-07-01 12:02:13 -05:00
|
|
|
package command
|
|
|
|
|
|
|
|
import (
|
|
|
|
"flag"
|
|
|
|
"fmt"
|
|
|
|
"strings"
|
2015-02-20 00:58:42 -06:00
|
|
|
|
2017-01-18 22:50:45 -06:00
|
|
|
"github.com/hashicorp/terraform/backend"
|
2017-05-01 16:47:53 -05:00
|
|
|
"github.com/hashicorp/terraform/config"
|
2017-01-18 22:50:45 -06:00
|
|
|
"github.com/hashicorp/terraform/config/module"
|
2016-11-09 08:58:52 -06:00
|
|
|
"github.com/hashicorp/terraform/dag"
|
2015-02-20 00:58:42 -06:00
|
|
|
"github.com/hashicorp/terraform/terraform"
|
2014-07-01 12:02:13 -05:00
|
|
|
)
|
|
|
|
|
|
|
|
// GraphCommand is a Command implementation that takes a Terraform
|
|
|
|
// configuration and outputs the dependency tree in graphical form.
|
|
|
|
type GraphCommand struct {
|
2014-07-12 22:37:30 -05:00
|
|
|
Meta
|
2014-07-01 12:02:13 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
func (c *GraphCommand) Run(args []string) int {
|
2014-09-24 19:36:27 -05:00
|
|
|
var moduleDepth int
|
2015-04-23 13:24:26 -05:00
|
|
|
var verbose bool
|
|
|
|
var drawCycles bool
|
2016-12-03 17:00:34 -06:00
|
|
|
var graphTypeStr string
|
2014-09-24 19:36:27 -05:00
|
|
|
|
2014-08-05 11:32:01 -05:00
|
|
|
args = c.Meta.process(args, false)
|
2014-07-12 22:37:30 -05:00
|
|
|
|
2014-07-01 12:02:13 -05:00
|
|
|
cmdFlags := flag.NewFlagSet("graph", flag.ContinueOnError)
|
2015-04-30 14:05:39 -05:00
|
|
|
c.addModuleDepthFlag(cmdFlags, &moduleDepth)
|
2015-04-23 13:24:26 -05:00
|
|
|
cmdFlags.BoolVar(&verbose, "verbose", false, "verbose")
|
|
|
|
cmdFlags.BoolVar(&drawCycles, "draw-cycles", false, "draw-cycles")
|
2016-12-03 17:00:34 -06:00
|
|
|
cmdFlags.StringVar(&graphTypeStr, "type", "", "type")
|
2014-07-01 12:02:13 -05:00
|
|
|
cmdFlags.Usage = func() { c.Ui.Error(c.Help()) }
|
|
|
|
if err := cmdFlags.Parse(args); err != nil {
|
|
|
|
return 1
|
|
|
|
}
|
|
|
|
|
2017-01-18 22:50:45 -06:00
|
|
|
configPath, err := ModulePath(cmdFlags.Args())
|
|
|
|
if err != nil {
|
|
|
|
c.Ui.Error(err.Error())
|
|
|
|
return 1
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check if the path is a plan
|
|
|
|
plan, err := c.Plan(configPath)
|
|
|
|
if err != nil {
|
|
|
|
c.Ui.Error(err.Error())
|
2014-07-01 12:02:13 -05:00
|
|
|
return 1
|
2017-01-18 22:50:45 -06:00
|
|
|
}
|
|
|
|
if plan != nil {
|
|
|
|
// Reset for backend loading
|
|
|
|
configPath = ""
|
|
|
|
}
|
|
|
|
|
|
|
|
// Load the module
|
|
|
|
var mod *module.Tree
|
|
|
|
if plan == nil {
|
|
|
|
mod, err = c.Module(configPath)
|
2014-07-11 22:41:47 -05:00
|
|
|
if err != nil {
|
2017-01-18 22:50:45 -06:00
|
|
|
c.Ui.Error(fmt.Sprintf("Failed to load root config module: %s", err))
|
|
|
|
return 1
|
2014-07-11 22:41:47 -05:00
|
|
|
}
|
2014-07-01 12:02:13 -05:00
|
|
|
}
|
|
|
|
|
2017-05-01 16:47:53 -05:00
|
|
|
var conf *config.Config
|
|
|
|
if mod != nil {
|
|
|
|
conf = mod.Config()
|
|
|
|
}
|
|
|
|
|
2017-01-18 22:50:45 -06:00
|
|
|
// Load the backend
|
|
|
|
b, err := c.Backend(&BackendOpts{
|
2017-05-01 16:47:53 -05:00
|
|
|
Config: conf,
|
|
|
|
Plan: plan,
|
2014-09-22 12:56:50 -05:00
|
|
|
})
|
2014-07-01 12:02:13 -05:00
|
|
|
if err != nil {
|
2017-01-18 22:50:45 -06:00
|
|
|
c.Ui.Error(fmt.Sprintf("Failed to load backend: %s", err))
|
|
|
|
return 1
|
|
|
|
}
|
|
|
|
|
|
|
|
// We require a local backend
|
|
|
|
local, ok := b.(backend.Local)
|
|
|
|
if !ok {
|
|
|
|
c.Ui.Error(ErrUnsupportedLocalOp)
|
|
|
|
return 1
|
|
|
|
}
|
|
|
|
|
|
|
|
// Build the operation
|
|
|
|
opReq := c.Operation()
|
|
|
|
opReq.Module = mod
|
|
|
|
opReq.Plan = plan
|
|
|
|
|
|
|
|
// Get the context
|
|
|
|
ctx, _, err := local.Context(opReq)
|
|
|
|
if err != nil {
|
|
|
|
c.Ui.Error(err.Error())
|
2014-07-01 12:02:13 -05:00
|
|
|
return 1
|
|
|
|
}
|
|
|
|
|
2016-12-03 17:00:34 -06:00
|
|
|
// Determine the graph type
|
|
|
|
graphType := terraform.GraphTypePlan
|
2017-01-18 22:50:45 -06:00
|
|
|
if plan != nil {
|
2016-12-03 17:00:34 -06:00
|
|
|
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 13:24:26 -05:00
|
|
|
// Skip validation during graph generation - we want to see the graph even if
|
|
|
|
// it is invalid for some reason.
|
2016-12-03 17:00:34 -06:00
|
|
|
g, err := ctx.Graph(graphType, &terraform.ContextGraphOpts{
|
2015-04-23 13:24:26 -05:00
|
|
|
Verbose: verbose,
|
|
|
|
Validate: false,
|
2015-04-23 10:52:31 -05:00
|
|
|
})
|
2014-07-01 12:02:13 -05:00
|
|
|
if err != nil {
|
|
|
|
c.Ui.Error(fmt.Sprintf("Error creating graph: %s", err))
|
|
|
|
return 1
|
|
|
|
}
|
|
|
|
|
2016-11-09 08:58:52 -06:00
|
|
|
graphStr, err := terraform.GraphDot(g, &dag.DotOpts{
|
2015-04-23 13:24:26 -05: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 12:02:13 -05:00
|
|
|
|
|
|
|
return 0
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *GraphCommand) Help() string {
|
|
|
|
helpText := `
|
2015-04-23 13:24:26 -05:00
|
|
|
Usage: terraform graph [options] [DIR]
|
2014-07-01 12:02:13 -05:00
|
|
|
|
2016-12-03 17:00:34 -06:00
|
|
|
Outputs the visual execution graph of Terraform resources according to
|
2015-04-23 13:24:26 -05:00
|
|
|
configuration files in DIR (or the current directory if omitted).
|
2014-07-01 12:02:13 -05:00
|
|
|
|
2014-07-11 22:38:03 -05: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-03 17:00:34 -06: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-24 19:36:27 -05:00
|
|
|
|
2016-12-03 17:00:34 -06:00
|
|
|
Options:
|
2015-04-23 13:24:26 -05:00
|
|
|
|
2016-12-03 17:00:34 -06:00
|
|
|
-draw-cycles Highlight any cycles in the graph with colored edges.
|
|
|
|
This helps when diagnosing cycle errors.
|
2014-09-24 19:36:27 -05:00
|
|
|
|
2016-12-03 17:00:34 -06:00
|
|
|
-no-color If specified, output won't contain any color.
|
2015-06-22 07:14:01 -05:00
|
|
|
|
2016-12-03 17:00:34 -06:00
|
|
|
-type=plan Type of graph to output. Can be: plan, plan-destroy, apply,
|
2017-01-26 17:18:42 -06:00
|
|
|
validate, input, refresh.
|
|
|
|
|
2015-06-22 07:14:01 -05:00
|
|
|
|
2014-07-01 12:02:13 -05:00
|
|
|
`
|
|
|
|
return strings.TrimSpace(helpText)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *GraphCommand) Synopsis() string {
|
2014-07-12 21:28:38 -05:00
|
|
|
return "Create a visual graph of Terraform resources"
|
2014-07-01 12:02:13 -05:00
|
|
|
}
|