pulumi/pkg/graph/dotconv/print.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()
}