diff options
Diffstat (limited to 'parser/dot.go')
| -rw-r--r-- | parser/dot.go | 74 |
1 files changed, 74 insertions, 0 deletions
diff --git a/parser/dot.go b/parser/dot.go new file mode 100644 index 0000000..cef1acb --- /dev/null +++ b/parser/dot.go @@ -0,0 +1,74 @@ +package parser + +import ( + "bytes" + "fmt" + "io" + "log" + "os/exec" + "strings" +) + +func (*Parser) Adot(nodes []*Node, c string) { + if c == "" { + return + } + n := &Node{Child: nodes} + n.Dot(c, "") +} + +func (n *Node) Dot(c, s string) { n.astDot(dotWriter(c), s) } + +func (n *Node) Sdot(s string) string { + var buf bytes.Buffer + n.astDot(&buf, s) + return buf.String() +} + +// TODO: rewrite it using Walk2 +func (n *Node) astDot(out io.Writer, label string) { + fmt.Fprintf(out, "digraph ast { ") + if label != "" { + fmt.Fprintf(out, "labelloc=\"t\"; label=\"%s\";", label) + } + anc := map[*Node]*Node{} + index := map[*Node]int{} + count := 0 + n.Walk(func(nod *Node) bool { + index[nod] = count + count++ + + for _, c := range nod.Child { + anc[c] = nod + } + name := strings.ReplaceAll(nod.Name(), `"`, `\"`) + fmt.Fprintf(out, "%d [label=\"%s\"]; ", index[nod], name) + if anc[nod] != nil { + fmt.Fprintf(out, "%d -> %d; ", index[anc[nod]], index[nod]) + } + return true + }, nil) + fmt.Fprintf(out, "}") +} + +type nopCloser struct { + io.Writer +} + +func (nopCloser) Close() error { return nil } + +func dotWriter(dotCmd string) io.WriteCloser { + if dotCmd == "" { + return nopCloser{io.Discard} + } + fields := strings.Fields(dotCmd) + cmd := exec.Command(fields[0], fields[1:]...) + dotin, err := cmd.StdinPipe() + if err != nil { + log.Fatal(err) + } + if err = cmd.Start(); err != nil { + log.Fatal(err) + } + return dotin +} |
