You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							327 lines
						
					
					
						
							7.1 KiB
						
					
					
				
			
		
		
	
	
							327 lines
						
					
					
						
							7.1 KiB
						
					
					
				| package pretty
 | |
| 
 | |
| import (
 | |
| 	"fmt"
 | |
| 	"io"
 | |
| 	"reflect"
 | |
| 	"strconv"
 | |
| 	"text/tabwriter"
 | |
| 
 | |
| 	"github.com/kr/text"
 | |
| )
 | |
| 
 | |
| type formatter struct {
 | |
| 	v     reflect.Value
 | |
| 	force bool
 | |
| 	quote bool
 | |
| }
 | |
| 
 | |
| // Formatter makes a wrapper, f, that will format x as go source with line
 | |
| // breaks and tabs. Object f responds to the "%v" formatting verb when both the
 | |
| // "#" and " " (space) flags are set, for example:
 | |
| //
 | |
| //     fmt.Sprintf("%# v", Formatter(x))
 | |
| //
 | |
| // If one of these two flags is not set, or any other verb is used, f will
 | |
| // format x according to the usual rules of package fmt.
 | |
| // In particular, if x satisfies fmt.Formatter, then x.Format will be called.
 | |
| func Formatter(x interface{}) (f fmt.Formatter) {
 | |
| 	return formatter{v: reflect.ValueOf(x), quote: true}
 | |
| }
 | |
| 
 | |
| func (fo formatter) String() string {
 | |
| 	return fmt.Sprint(fo.v.Interface()) // unwrap it
 | |
| }
 | |
| 
 | |
| func (fo formatter) passThrough(f fmt.State, c rune) {
 | |
| 	s := "%"
 | |
| 	for i := 0; i < 128; i++ {
 | |
| 		if f.Flag(i) {
 | |
| 			s += string(i)
 | |
| 		}
 | |
| 	}
 | |
| 	if w, ok := f.Width(); ok {
 | |
| 		s += fmt.Sprintf("%d", w)
 | |
| 	}
 | |
| 	if p, ok := f.Precision(); ok {
 | |
| 		s += fmt.Sprintf(".%d", p)
 | |
| 	}
 | |
| 	s += string(c)
 | |
| 	fmt.Fprintf(f, s, fo.v.Interface())
 | |
| }
 | |
| 
 | |
| func (fo formatter) Format(f fmt.State, c rune) {
 | |
| 	if fo.force || c == 'v' && f.Flag('#') && f.Flag(' ') {
 | |
| 		w := tabwriter.NewWriter(f, 4, 4, 1, ' ', 0)
 | |
| 		p := &printer{tw: w, Writer: w, visited: make(map[visit]int)}
 | |
| 		p.printValue(fo.v, true, fo.quote)
 | |
| 		w.Flush()
 | |
| 		return
 | |
| 	}
 | |
| 	fo.passThrough(f, c)
 | |
| }
 | |
| 
 | |
| type printer struct {
 | |
| 	io.Writer
 | |
| 	tw      *tabwriter.Writer
 | |
| 	visited map[visit]int
 | |
| 	depth   int
 | |
| }
 | |
| 
 | |
| func (p *printer) indent() *printer {
 | |
| 	q := *p
 | |
| 	q.tw = tabwriter.NewWriter(p.Writer, 4, 4, 1, ' ', 0)
 | |
| 	q.Writer = text.NewIndentWriter(q.tw, []byte{'\t'})
 | |
| 	return &q
 | |
| }
 | |
| 
 | |
| func (p *printer) printInline(v reflect.Value, x interface{}, showType bool) {
 | |
| 	if showType {
 | |
| 		io.WriteString(p, v.Type().String())
 | |
| 		fmt.Fprintf(p, "(%#v)", x)
 | |
| 	} else {
 | |
| 		fmt.Fprintf(p, "%#v", x)
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // printValue must keep track of already-printed pointer values to avoid
 | |
| // infinite recursion.
 | |
| type visit struct {
 | |
| 	v   uintptr
 | |
| 	typ reflect.Type
 | |
| }
 | |
| 
 | |
| func (p *printer) printValue(v reflect.Value, showType, quote bool) {
 | |
| 	if p.depth > 10 {
 | |
| 		io.WriteString(p, "!%v(DEPTH EXCEEDED)")
 | |
| 		return
 | |
| 	}
 | |
| 
 | |
| 	switch v.Kind() {
 | |
| 	case reflect.Bool:
 | |
| 		p.printInline(v, v.Bool(), showType)
 | |
| 	case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:
 | |
| 		p.printInline(v, v.Int(), showType)
 | |
| 	case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Uintptr:
 | |
| 		p.printInline(v, v.Uint(), showType)
 | |
| 	case reflect.Float32, reflect.Float64:
 | |
| 		p.printInline(v, v.Float(), showType)
 | |
| 	case reflect.Complex64, reflect.Complex128:
 | |
| 		fmt.Fprintf(p, "%#v", v.Complex())
 | |
| 	case reflect.String:
 | |
| 		p.fmtString(v.String(), quote)
 | |
| 	case reflect.Map:
 | |
| 		t := v.Type()
 | |
| 		if showType {
 | |
| 			io.WriteString(p, t.String())
 | |
| 		}
 | |
| 		writeByte(p, '{')
 | |
| 		if nonzero(v) {
 | |
| 			expand := !canInline(v.Type())
 | |
| 			pp := p
 | |
| 			if expand {
 | |
| 				writeByte(p, '\n')
 | |
| 				pp = p.indent()
 | |
| 			}
 | |
| 			keys := v.MapKeys()
 | |
| 			for i := 0; i < v.Len(); i++ {
 | |
| 				k := keys[i]
 | |
| 				mv := v.MapIndex(k)
 | |
| 				pp.printValue(k, false, true)
 | |
| 				writeByte(pp, ':')
 | |
| 				if expand {
 | |
| 					writeByte(pp, '\t')
 | |
| 				}
 | |
| 				showTypeInStruct := t.Elem().Kind() == reflect.Interface
 | |
| 				pp.printValue(mv, showTypeInStruct, true)
 | |
| 				if expand {
 | |
| 					io.WriteString(pp, ",\n")
 | |
| 				} else if i < v.Len()-1 {
 | |
| 					io.WriteString(pp, ", ")
 | |
| 				}
 | |
| 			}
 | |
| 			if expand {
 | |
| 				pp.tw.Flush()
 | |
| 			}
 | |
| 		}
 | |
| 		writeByte(p, '}')
 | |
| 	case reflect.Struct:
 | |
| 		t := v.Type()
 | |
| 		if v.CanAddr() {
 | |
| 			addr := v.UnsafeAddr()
 | |
| 			vis := visit{addr, t}
 | |
| 			if vd, ok := p.visited[vis]; ok && vd < p.depth {
 | |
| 				p.fmtString(t.String()+"{(CYCLIC REFERENCE)}", false)
 | |
| 				break // don't print v again
 | |
| 			}
 | |
| 			p.visited[vis] = p.depth
 | |
| 		}
 | |
| 
 | |
| 		if showType {
 | |
| 			io.WriteString(p, t.String())
 | |
| 		}
 | |
| 		writeByte(p, '{')
 | |
| 		if nonzero(v) {
 | |
| 			expand := !canInline(v.Type())
 | |
| 			pp := p
 | |
| 			if expand {
 | |
| 				writeByte(p, '\n')
 | |
| 				pp = p.indent()
 | |
| 			}
 | |
| 			for i := 0; i < v.NumField(); i++ {
 | |
| 				showTypeInStruct := true
 | |
| 				if f := t.Field(i); f.Name != "" {
 | |
| 					io.WriteString(pp, f.Name)
 | |
| 					writeByte(pp, ':')
 | |
| 					if expand {
 | |
| 						writeByte(pp, '\t')
 | |
| 					}
 | |
| 					showTypeInStruct = labelType(f.Type)
 | |
| 				}
 | |
| 				pp.printValue(getField(v, i), showTypeInStruct, true)
 | |
| 				if expand {
 | |
| 					io.WriteString(pp, ",\n")
 | |
| 				} else if i < v.NumField()-1 {
 | |
| 					io.WriteString(pp, ", ")
 | |
| 				}
 | |
| 			}
 | |
| 			if expand {
 | |
| 				pp.tw.Flush()
 | |
| 			}
 | |
| 		}
 | |
| 		writeByte(p, '}')
 | |
| 	case reflect.Interface:
 | |
| 		switch e := v.Elem(); {
 | |
| 		case e.Kind() == reflect.Invalid:
 | |
| 			io.WriteString(p, "nil")
 | |
| 		case e.IsValid():
 | |
| 			pp := *p
 | |
| 			pp.depth++
 | |
| 			pp.printValue(e, showType, true)
 | |
| 		default:
 | |
| 			io.WriteString(p, v.Type().String())
 | |
| 			io.WriteString(p, "(nil)")
 | |
| 		}
 | |
| 	case reflect.Array, reflect.Slice:
 | |
| 		t := v.Type()
 | |
| 		if showType {
 | |
| 			io.WriteString(p, t.String())
 | |
| 		}
 | |
| 		if v.Kind() == reflect.Slice && v.IsNil() && showType {
 | |
| 			io.WriteString(p, "(nil)")
 | |
| 			break
 | |
| 		}
 | |
| 		if v.Kind() == reflect.Slice && v.IsNil() {
 | |
| 			io.WriteString(p, "nil")
 | |
| 			break
 | |
| 		}
 | |
| 		writeByte(p, '{')
 | |
| 		expand := !canInline(v.Type())
 | |
| 		pp := p
 | |
| 		if expand {
 | |
| 			writeByte(p, '\n')
 | |
| 			pp = p.indent()
 | |
| 		}
 | |
| 		for i := 0; i < v.Len(); i++ {
 | |
| 			showTypeInSlice := t.Elem().Kind() == reflect.Interface
 | |
| 			pp.printValue(v.Index(i), showTypeInSlice, true)
 | |
| 			if expand {
 | |
| 				io.WriteString(pp, ",\n")
 | |
| 			} else if i < v.Len()-1 {
 | |
| 				io.WriteString(pp, ", ")
 | |
| 			}
 | |
| 		}
 | |
| 		if expand {
 | |
| 			pp.tw.Flush()
 | |
| 		}
 | |
| 		writeByte(p, '}')
 | |
| 	case reflect.Ptr:
 | |
| 		e := v.Elem()
 | |
| 		if !e.IsValid() {
 | |
| 			writeByte(p, '(')
 | |
| 			io.WriteString(p, v.Type().String())
 | |
| 			io.WriteString(p, ")(nil)")
 | |
| 		} else {
 | |
| 			pp := *p
 | |
| 			pp.depth++
 | |
| 			writeByte(pp, '&')
 | |
| 			pp.printValue(e, true, true)
 | |
| 		}
 | |
| 	case reflect.Chan:
 | |
| 		x := v.Pointer()
 | |
| 		if showType {
 | |
| 			writeByte(p, '(')
 | |
| 			io.WriteString(p, v.Type().String())
 | |
| 			fmt.Fprintf(p, ")(%#v)", x)
 | |
| 		} else {
 | |
| 			fmt.Fprintf(p, "%#v", x)
 | |
| 		}
 | |
| 	case reflect.Func:
 | |
| 		io.WriteString(p, v.Type().String())
 | |
| 		io.WriteString(p, " {...}")
 | |
| 	case reflect.UnsafePointer:
 | |
| 		p.printInline(v, v.Pointer(), showType)
 | |
| 	case reflect.Invalid:
 | |
| 		io.WriteString(p, "nil")
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func canInline(t reflect.Type) bool {
 | |
| 	switch t.Kind() {
 | |
| 	case reflect.Map:
 | |
| 		return !canExpand(t.Elem())
 | |
| 	case reflect.Struct:
 | |
| 		for i := 0; i < t.NumField(); i++ {
 | |
| 			if canExpand(t.Field(i).Type) {
 | |
| 				return false
 | |
| 			}
 | |
| 		}
 | |
| 		return true
 | |
| 	case reflect.Interface:
 | |
| 		return false
 | |
| 	case reflect.Array, reflect.Slice:
 | |
| 		return !canExpand(t.Elem())
 | |
| 	case reflect.Ptr:
 | |
| 		return false
 | |
| 	case reflect.Chan, reflect.Func, reflect.UnsafePointer:
 | |
| 		return false
 | |
| 	}
 | |
| 	return true
 | |
| }
 | |
| 
 | |
| func canExpand(t reflect.Type) bool {
 | |
| 	switch t.Kind() {
 | |
| 	case reflect.Map, reflect.Struct,
 | |
| 		reflect.Interface, reflect.Array, reflect.Slice,
 | |
| 		reflect.Ptr:
 | |
| 		return true
 | |
| 	}
 | |
| 	return false
 | |
| }
 | |
| 
 | |
| func labelType(t reflect.Type) bool {
 | |
| 	switch t.Kind() {
 | |
| 	case reflect.Interface, reflect.Struct:
 | |
| 		return true
 | |
| 	}
 | |
| 	return false
 | |
| }
 | |
| 
 | |
| func (p *printer) fmtString(s string, quote bool) {
 | |
| 	if quote {
 | |
| 		s = strconv.Quote(s)
 | |
| 	}
 | |
| 	io.WriteString(p, s)
 | |
| }
 | |
| 
 | |
| func writeByte(w io.Writer, b byte) {
 | |
| 	w.Write([]byte{b})
 | |
| }
 | |
| 
 | |
| func getField(v reflect.Value, i int) reflect.Value {
 | |
| 	val := v.Field(i)
 | |
| 	if val.Kind() == reflect.Interface && !val.IsNil() {
 | |
| 		val = val.Elem()
 | |
| 	}
 | |
| 	return val
 | |
| }
 | |
| 
 |