/* * [The "BSD licence"] * Copyright (c) 2005-2008 Terence Parr * All rights reserved. * * Conversion to C#: * Copyright (c) 2008 Sam Harwell, Pixel Mine, Inc. * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * 3. The name of the author may not be used to endorse or promote products * derived from this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ namespace Antlr.Runtime.Tree { using System.Collections.Generic; using StringBuilder = System.Text.StringBuilder; /** A utility class to generate DOT diagrams (graphviz) from * arbitrary trees. You can pass in your own templates and * can pass in any kind of tree or use Tree interface method. * I wanted this separator so that you don't have to include * ST just to use the org.antlr.runtime.tree.* package. * This is a set of non-static methods so you can subclass * to override. For example, here is an invocation: * * CharStream input = new ANTLRInputStream(System.in); * TLexer lex = new TLexer(input); * CommonTokenStream tokens = new CommonTokenStream(lex); * TParser parser = new TParser(tokens); * TParser.e_return r = parser.e(); * Tree t = (Tree)r.tree; * System.out.println(t.toStringTree()); * DOTTreeGenerator gen = new DOTTreeGenerator(); * StringTemplate st = gen.toDOT(t); * System.out.println(st); */ public class DotTreeGenerator { readonly string[] HeaderLines = { "digraph {", "", "\tordering=out;", "\tranksep=.4;", "\tbgcolor=\"lightgrey\"; node [shape=box, fixedsize=false, fontsize=12, fontname=\"Helvetica-bold\", fontcolor=\"blue\"", "\t\twidth=.25, height=.25, color=\"black\", fillcolor=\"white\", style=\"filled, solid, bold\"];", "\tedge [arrowsize=.5, color=\"black\", style=\"bold\"]", "" }; const string Footer = "}"; const string NodeFormat = " {0} [label=\"{1}\"];"; const string EdgeFormat = " {0} -> {1} // \"{2}\" -> \"{3}\""; /** Track node to number mapping so we can get proper node name back */ Dictionary nodeToNumberMap = new Dictionary(); /** Track node number so we can get unique node names */ int nodeNumber = 0; /** Generate DOT (graphviz) for a whole tree not just a node. * For example, 3+4*5 should generate: * * digraph { * node [shape=plaintext, fixedsize=true, fontsize=11, fontname="Courier", * width=.4, height=.2]; * edge [arrowsize=.7] * "+"->3 * "+"->"*" * "*"->4 * "*"->5 * } * * Takes a Tree interface object. */ public virtual string ToDot( object tree, ITreeAdaptor adaptor ) { StringBuilder builder = new StringBuilder(); foreach ( string line in HeaderLines ) builder.AppendLine( line ); nodeNumber = 0; var nodes = DefineNodes( tree, adaptor ); nodeNumber = 0; var edges = DefineEdges( tree, adaptor ); foreach ( var s in nodes ) builder.AppendLine( s ); builder.AppendLine(); foreach ( var s in edges ) builder.AppendLine( s ); builder.AppendLine(); builder.AppendLine( Footer ); return builder.ToString(); } public virtual string ToDot( ITree tree ) { return ToDot( tree, new CommonTreeAdaptor() ); } protected virtual IEnumerable DefineNodes( object tree, ITreeAdaptor adaptor ) { if ( tree == null ) yield break; int n = adaptor.GetChildCount( tree ); if ( n == 0 ) { // must have already dumped as child from previous // invocation; do nothing yield break; } // define parent node yield return GetNodeText( adaptor, tree ); // for each child, do a " [label=text]" node def for ( int i = 0; i < n; i++ ) { object child = adaptor.GetChild( tree, i ); yield return GetNodeText( adaptor, child ); foreach ( var t in DefineNodes( child, adaptor ) ) yield return t; } } protected virtual IEnumerable DefineEdges( object tree, ITreeAdaptor adaptor ) { if ( tree == null ) yield break; int n = adaptor.GetChildCount( tree ); if ( n == 0 ) { // must have already dumped as child from previous // invocation; do nothing yield break; } string parentName = "n" + GetNodeNumber( tree ); // for each child, do a parent -> child edge using unique node names string parentText = adaptor.GetText( tree ); for ( int i = 0; i < n; i++ ) { object child = adaptor.GetChild( tree, i ); string childText = adaptor.GetText( child ); string childName = "n" + GetNodeNumber( child ); yield return string.Format( EdgeFormat, parentName, childName, FixString( parentText ), FixString( childText ) ); foreach ( var t in DefineEdges( child, adaptor ) ) yield return t; } } protected virtual string GetNodeText( ITreeAdaptor adaptor, object t ) { string text = adaptor.GetText( t ); string uniqueName = "n" + GetNodeNumber( t ); return string.Format( NodeFormat, uniqueName, FixString( text ) ); } protected virtual int GetNodeNumber( object t ) { int i; if ( nodeToNumberMap.TryGetValue( t, out i ) ) { return i; } else { nodeToNumberMap[t] = nodeNumber; nodeNumber++; return nodeNumber - 1; } } protected virtual string FixString( string text ) { if ( text != null ) { text = System.Text.RegularExpressions.Regex.Replace( text, "\"", "\\\\\"" ); text = System.Text.RegularExpressions.Regex.Replace( text, "\\t", " " ); text = System.Text.RegularExpressions.Regex.Replace( text, "\\n", "\\\\n" ); text = System.Text.RegularExpressions.Regex.Replace( text, "\\r", "\\\\r" ); if ( text.Length > 20 ) text = text.Substring( 0, 8 ) + "..." + text.Substring( text.Length - 8 ); } return text; } } }