aboutsummaryrefslogtreecommitdiffstats
path: root/core/vm/sqlvm/ast/printer.go
blob: 4d977aeea70257965dc3481335fa528165377672 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
package ast

import (
    "fmt"
    "reflect"
    "strconv"
    "unicode"
    "unicode/utf8"
)

func formatBytes(b []byte) string {
    if utf8.Valid(b) {
        for r, i, size := rune(0), 0, 0; i < len(b); i += size {
            r, size = utf8.DecodeRune(b[i:])
            if !unicode.IsPrint(r) {
                return strconv.Quote(string(b))
            }
        }
        return string(b)
    }
    return fmt.Sprintf("%v", b)
}

func formatString(s string) string {
    if utf8.ValidString(s) {
        for _, r := range s {
            if !unicode.IsPrint(r) {
                return strconv.Quote(s)
            }
        }
        return s
    }
    return fmt.Sprintf("%v", []byte(s))
}

// PrintAST prints ast to stdout.
func PrintAST(n interface{}, indent string, detail bool) {
    if n == nil {
        fmt.Printf("%snil\n", indent)
        return
    }
    typeOf := reflect.TypeOf(n)
    valueOf := reflect.ValueOf(n)
    name := ""
    if typeOf.Kind() == reflect.Ptr {
        if valueOf.IsNil() {
            fmt.Printf("%snil\n", indent)
            return
        }
        name = "*"
        valueOf = valueOf.Elem()
        typeOf = typeOf.Elem()
    }
    name = name + typeOf.Name()

    if op, ok := n.(UnaryOperator); ok {
        fmt.Printf("%s%s:\n", indent, name)
        fmt.Printf("%s  Target:\n", indent)
        PrintAST(op.GetTarget(), indent+"    ", detail)
        return
    }
    if op, ok := n.(BinaryOperator); ok {
        fmt.Printf("%s%s:\n", indent, name)
        fmt.Printf("%s  Object:\n", indent)
        PrintAST(op.GetObject(), indent+"    ", detail)
        fmt.Printf("%s  Subject:\n", indent)
        PrintAST(op.GetSubject(), indent+"    ", detail)
        return
    }
    if arr, ok := n.([]interface{}); ok {
        if len(arr) == 0 {
            fmt.Printf("%s[]\n", indent)
            return
        }
        fmt.Printf("%s[\n", indent)
        for idx := range arr {
            PrintAST(arr[idx], indent+"  ", detail)
        }
        fmt.Printf("%s]\n", indent)
        return
    }
    if stringer, ok := n.(fmt.Stringer); ok {
        s := stringer.String()
        fmt.Printf("%s%s: %s\n", indent, name, formatString(s))
        return
    }
    if typeOf.Kind() == reflect.Struct {
        fmt.Printf("%s%s", indent, name)
        l := typeOf.NumField()
        if l == 0 {
            fmt.Printf(" {}\n")
            return
        }
        fmt.Printf(" {\n")
        for i := 0; i < l; i++ {
            if !detail && typeOf.Field(i).Tag.Get("print") == "-" {
                continue
            }
            fmt.Printf("%s  %s:\n", indent, typeOf.Field(i).Name)
            PrintAST(valueOf.Field(i).Interface(), indent+"    ", detail)
        }
        fmt.Printf("%s}\n", indent)
        return
    }
    if bs, ok := n.([]byte); ok {
        fmt.Printf("%s%s\n", indent, formatBytes(bs))
        return
    }
    fmt.Printf("%s%+v\n", indent, valueOf.Interface())
}