mirror of https://github.com/pulumi/pulumi.git
148 lines
4.3 KiB
Go
148 lines
4.3 KiB
Go
// Copyright 2016-2018, Pulumi Corporation.
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
// Package dotconv converts a resource graph into its DOT digraph equivalent. This is useful for integration with
|
|
// various visualization tools, like Graphviz. Please see http://www.graphviz.org/content/dot-language for a thorough
|
|
// specification of the DOT file format.
|
|
package dotconv
|
|
|
|
import (
|
|
"bufio"
|
|
"fmt"
|
|
"io"
|
|
"strconv"
|
|
"strings"
|
|
|
|
"github.com/pulumi/pulumi/pkg/v3/graph"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/slice"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/util/contract"
|
|
)
|
|
|
|
// Print prints a resource graph.
|
|
func Print(g graph.Graph, w io.Writer, dotFragment string) error {
|
|
// Allocate a new writer. In general, we will ignore write errors throughout this function, for simplicity, opting
|
|
// instead to return the result of flushing the buffer at the end, which is generally latching.
|
|
b := bufio.NewWriter(w)
|
|
|
|
// Print the graph header.
|
|
if _, err := b.WriteString("strict digraph {\n"); err != nil {
|
|
return err
|
|
}
|
|
|
|
// If the caller provided a fragment then insert it here.
|
|
if dotFragment != "" {
|
|
if _, err := b.WriteString(dotFragment); err != nil {
|
|
return err
|
|
}
|
|
|
|
// Ensure that the fragment is followed by newline, this reduces
|
|
// problems if the fragment doesn't end with a semicolon
|
|
if _, err := b.WriteString("\n"); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
// Initialize the frontier with unvisited graph vertices.
|
|
queued := make(map[graph.Vertex]bool)
|
|
frontier := slice.Prealloc[graph.Vertex](len(g.Roots()))
|
|
for _, root := range g.Roots() {
|
|
to := root.To()
|
|
queued[to] = true
|
|
frontier = append(frontier, to)
|
|
}
|
|
|
|
// For now, we auto-generate IDs.
|
|
// TODO[pulumi/pulumi#76]: use the object URNs instead, once we have them.
|
|
c := 0
|
|
ids := make(map[graph.Vertex]string)
|
|
getID := func(v graph.Vertex) string {
|
|
if id, has := ids[v]; has {
|
|
return id
|
|
}
|
|
id := "Resource" + strconv.Itoa(c)
|
|
c++
|
|
ids[v] = id
|
|
return id
|
|
}
|
|
|
|
// Now, until the frontier is empty, emit entries into the stream.
|
|
indent := " "
|
|
emitted := make(map[graph.Vertex]bool)
|
|
for len(frontier) > 0 {
|
|
// Dequeue the head of the frontier.
|
|
v := frontier[0]
|
|
frontier = frontier[1:]
|
|
contract.Assertf(!emitted[v], "vertex was emitted twice")
|
|
emitted[v] = true
|
|
|
|
// Get and lazily allocate the ID for this vertex.
|
|
id := getID(v)
|
|
|
|
// Print this vertex; first its "label" (type) and then its direct dependencies.
|
|
// IDEA: consider serializing properties on the node also.
|
|
if _, err := fmt.Fprintf(b, "%v%v", indent, id); err != nil {
|
|
return err
|
|
}
|
|
if label := v.Label(); label != "" {
|
|
if _, err := fmt.Fprintf(b, " [label=\"%v\"]", label); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
if _, err := b.WriteString(";\n"); err != nil {
|
|
return err
|
|
}
|
|
|
|
// Now print out all dependencies as "ID -> {A ... Z}".
|
|
outs := v.Outs()
|
|
if len(outs) > 0 {
|
|
base := fmt.Sprintf("%v%v", indent, id)
|
|
// Print the ID of each dependency and, for those we haven't seen, add them to the frontier.
|
|
for _, out := range outs {
|
|
to := out.To()
|
|
if _, err := fmt.Fprintf(b, "%s -> %s", base, getID(to)); err != nil {
|
|
return err
|
|
}
|
|
|
|
var attrs []string
|
|
if out.Color() != "" {
|
|
attrs = append(attrs, fmt.Sprintf("color = \"%s\"", out.Color()))
|
|
}
|
|
if out.Label() != "" {
|
|
attrs = append(attrs, fmt.Sprintf("label = \"%s\"", out.Label()))
|
|
}
|
|
if len(attrs) > 0 {
|
|
if _, err := fmt.Fprintf(b, " [%s]", strings.Join(attrs, ", ")); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
if _, err := b.WriteString(";\n"); err != nil {
|
|
return err
|
|
}
|
|
|
|
if _, q := queued[to]; !q {
|
|
queued[to] = true
|
|
frontier = append(frontier, to)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// Finish the graph.
|
|
if _, err := b.WriteString("}\n"); err != nil {
|
|
return err
|
|
}
|
|
return b.Flush()
|
|
}
|