github.com/v2pro/plz@v0.0.0-20221028024117-e5f9aec5b631/test/go-spew/spew/format.go (about)

     1  /*
     2   * Copyright (c) 2013-2016 Dave Collins <dave@davec.name>
     3   *
     4   * Permission to use, copy, modify, and distribute this software for any
     5   * purpose with or without fee is hereby granted, provided that the above
     6   * copyright notice and this permission notice appear in all copies.
     7   *
     8   * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
     9   * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
    10   * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
    11   * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
    12   * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
    13   * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
    14   * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
    15   */
    16  
    17  package spew
    18  
    19  import (
    20  	"bytes"
    21  	"fmt"
    22  	"reflect"
    23  	"strconv"
    24  	"strings"
    25  )
    26  
    27  // supportedFlags is a list of all the character flags supported by fmt package.
    28  const supportedFlags = "0-+# "
    29  
    30  // formatState implements the fmt.Formatter interface and contains information
    31  // about the state of a formatting operation.  The NewFormatter function can
    32  // be used to get a new Formatter which can be used directly as arguments
    33  // in standard fmt package printing calls.
    34  type formatState struct {
    35  	value          interface{}
    36  	fs             fmt.State
    37  	depth          int
    38  	pointers       map[uintptr]int
    39  	ignoreNextType bool
    40  	cs             *ConfigState
    41  }
    42  
    43  // buildDefaultFormat recreates the original format string without precision
    44  // and width information to pass in to fmt.Sprintf in the case of an
    45  // unrecognized type.  Unless new types are added to the language, this
    46  // function won't ever be called.
    47  func (f *formatState) buildDefaultFormat() (format string) {
    48  	buf := bytes.NewBuffer(percentBytes)
    49  
    50  	for _, flag := range supportedFlags {
    51  		if f.fs.Flag(int(flag)) {
    52  			buf.WriteRune(flag)
    53  		}
    54  	}
    55  
    56  	buf.WriteRune('v')
    57  
    58  	format = buf.String()
    59  	return format
    60  }
    61  
    62  // constructOrigFormat recreates the original format string including precision
    63  // and width information to pass along to the standard fmt package.  This allows
    64  // automatic deferral of all format strings this package doesn't support.
    65  func (f *formatState) constructOrigFormat(verb rune) (format string) {
    66  	buf := bytes.NewBuffer(percentBytes)
    67  
    68  	for _, flag := range supportedFlags {
    69  		if f.fs.Flag(int(flag)) {
    70  			buf.WriteRune(flag)
    71  		}
    72  	}
    73  
    74  	if width, ok := f.fs.Width(); ok {
    75  		buf.WriteString(strconv.Itoa(width))
    76  	}
    77  
    78  	if precision, ok := f.fs.Precision(); ok {
    79  		buf.Write(precisionBytes)
    80  		buf.WriteString(strconv.Itoa(precision))
    81  	}
    82  
    83  	buf.WriteRune(verb)
    84  
    85  	format = buf.String()
    86  	return format
    87  }
    88  
    89  // unpackValue returns values inside of non-nil interfaces when possible and
    90  // ensures that types for values which have been unpacked from an interface
    91  // are displayed when the show types flag is also set.
    92  // This is useful for data types like structs, arrays, slices, and maps which
    93  // can contain varying types packed inside an interface.
    94  func (f *formatState) unpackValue(v reflect.Value) reflect.Value {
    95  	if v.Kind() == reflect.Interface {
    96  		f.ignoreNextType = false
    97  		if !v.IsNil() {
    98  			v = v.Elem()
    99  		}
   100  	}
   101  	return v
   102  }
   103  
   104  // formatPtr handles formatting of pointers by indirecting them as necessary.
   105  func (f *formatState) formatPtr(v reflect.Value) {
   106  	// Display nil if top level pointer is nil.
   107  	showTypes := f.fs.Flag('#')
   108  	if v.IsNil() && (!showTypes || f.ignoreNextType) {
   109  		f.fs.Write(nilAngleBytes)
   110  		return
   111  	}
   112  
   113  	// Remove pointers at or below the current depth from map used to detect
   114  	// circular refs.
   115  	for k, depth := range f.pointers {
   116  		if depth >= f.depth {
   117  			delete(f.pointers, k)
   118  		}
   119  	}
   120  
   121  	// Keep list of all dereferenced pointers to possibly show later.
   122  	pointerChain := make([]uintptr, 0)
   123  
   124  	// Figure out how many levels of indirection there are by derferencing
   125  	// pointers and unpacking interfaces down the chain while detecting circular
   126  	// references.
   127  	nilFound := false
   128  	cycleFound := false
   129  	indirects := 0
   130  	ve := v
   131  	for ve.Kind() == reflect.Ptr {
   132  		if ve.IsNil() {
   133  			nilFound = true
   134  			break
   135  		}
   136  		indirects++
   137  		addr := ve.Pointer()
   138  		pointerChain = append(pointerChain, addr)
   139  		if pd, ok := f.pointers[addr]; ok && pd < f.depth {
   140  			cycleFound = true
   141  			indirects--
   142  			break
   143  		}
   144  		f.pointers[addr] = f.depth
   145  
   146  		ve = ve.Elem()
   147  		if ve.Kind() == reflect.Interface {
   148  			if ve.IsNil() {
   149  				nilFound = true
   150  				break
   151  			}
   152  			ve = ve.Elem()
   153  		}
   154  	}
   155  
   156  	// Display type or indirection level depending on flags.
   157  	if showTypes && !f.ignoreNextType {
   158  		f.fs.Write(openParenBytes)
   159  		f.fs.Write(bytes.Repeat(asteriskBytes, indirects))
   160  		f.fs.Write([]byte(ve.Type().String()))
   161  		f.fs.Write(closeParenBytes)
   162  	} else {
   163  		if nilFound || cycleFound {
   164  			indirects += strings.Count(ve.Type().String(), "*")
   165  		}
   166  		f.fs.Write(openAngleBytes)
   167  		f.fs.Write([]byte(strings.Repeat("*", indirects)))
   168  		f.fs.Write(closeAngleBytes)
   169  	}
   170  
   171  	// Display pointer information depending on flags.
   172  	if f.fs.Flag('+') && (len(pointerChain) > 0) {
   173  		f.fs.Write(openParenBytes)
   174  		for i, addr := range pointerChain {
   175  			if i > 0 {
   176  				f.fs.Write(pointerChainBytes)
   177  			}
   178  			printHexPtr(f.fs, addr)
   179  		}
   180  		f.fs.Write(closeParenBytes)
   181  	}
   182  
   183  	// Display dereferenced value.
   184  	switch {
   185  	case nilFound == true:
   186  		f.fs.Write(nilAngleBytes)
   187  
   188  	case cycleFound == true:
   189  		f.fs.Write(circularShortBytes)
   190  
   191  	default:
   192  		f.ignoreNextType = true
   193  		f.format(ve)
   194  	}
   195  }
   196  
   197  // format is the main workhorse for providing the Formatter interface.  It
   198  // uses the passed reflect value to figure out what kind of object we are
   199  // dealing with and formats it appropriately.  It is a recursive function,
   200  // however circular data structures are detected and handled properly.
   201  func (f *formatState) format(v reflect.Value) {
   202  	// Handle invalid reflect values immediately.
   203  	kind := v.Kind()
   204  	if kind == reflect.Invalid {
   205  		f.fs.Write(invalidAngleBytes)
   206  		return
   207  	}
   208  
   209  	// Handle pointers specially.
   210  	if kind == reflect.Ptr {
   211  		f.formatPtr(v)
   212  		return
   213  	}
   214  
   215  	// Print type information unless already handled elsewhere.
   216  	if !f.ignoreNextType && f.fs.Flag('#') {
   217  		f.fs.Write(openParenBytes)
   218  		f.fs.Write([]byte(v.Type().String()))
   219  		f.fs.Write(closeParenBytes)
   220  	}
   221  	f.ignoreNextType = false
   222  
   223  	// Call Stringer/error interfaces if they exist and the handle methods
   224  	// flag is enabled.
   225  	if !f.cs.DisableMethods {
   226  		if (kind != reflect.Invalid) && (kind != reflect.Interface) {
   227  			if handled := handleMethods(f.cs, f.fs, v); handled {
   228  				return
   229  			}
   230  		}
   231  	}
   232  
   233  	switch kind {
   234  	case reflect.Invalid:
   235  		// Do nothing.  We should never get here since invalid has already
   236  		// been handled above.
   237  
   238  	case reflect.Bool:
   239  		printBool(f.fs, v.Bool())
   240  
   241  	case reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64, reflect.Int:
   242  		printInt(f.fs, v.Int(), 10)
   243  
   244  	case reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Uint:
   245  		printUint(f.fs, v.Uint(), 10)
   246  
   247  	case reflect.Float32:
   248  		printFloat(f.fs, v.Float(), 32)
   249  
   250  	case reflect.Float64:
   251  		printFloat(f.fs, v.Float(), 64)
   252  
   253  	case reflect.Complex64:
   254  		printComplex(f.fs, v.Complex(), 32)
   255  
   256  	case reflect.Complex128:
   257  		printComplex(f.fs, v.Complex(), 64)
   258  
   259  	case reflect.Slice:
   260  		if v.IsNil() {
   261  			f.fs.Write(nilAngleBytes)
   262  			break
   263  		}
   264  		fallthrough
   265  
   266  	case reflect.Array:
   267  		f.fs.Write(openBracketBytes)
   268  		f.depth++
   269  		if (f.cs.MaxDepth != 0) && (f.depth > f.cs.MaxDepth) {
   270  			f.fs.Write(maxShortBytes)
   271  		} else {
   272  			numEntries := v.Len()
   273  			for i := 0; i < numEntries; i++ {
   274  				if i > 0 {
   275  					f.fs.Write(spaceBytes)
   276  				}
   277  				f.ignoreNextType = true
   278  				f.format(f.unpackValue(v.Index(i)))
   279  			}
   280  		}
   281  		f.depth--
   282  		f.fs.Write(closeBracketBytes)
   283  
   284  	case reflect.String:
   285  		f.fs.Write([]byte(v.String()))
   286  
   287  	case reflect.Interface:
   288  		// The only time we should get here is for nil interfaces due to
   289  		// unpackValue calls.
   290  		if v.IsNil() {
   291  			f.fs.Write(nilAngleBytes)
   292  		}
   293  
   294  	case reflect.Ptr:
   295  		// Do nothing.  We should never get here since pointers have already
   296  		// been handled above.
   297  
   298  	case reflect.Map:
   299  		// nil maps should be indicated as different than empty maps
   300  		if v.IsNil() {
   301  			f.fs.Write(nilAngleBytes)
   302  			break
   303  		}
   304  
   305  		f.fs.Write(openMapBytes)
   306  		f.depth++
   307  		if (f.cs.MaxDepth != 0) && (f.depth > f.cs.MaxDepth) {
   308  			f.fs.Write(maxShortBytes)
   309  		} else {
   310  			keys := v.MapKeys()
   311  			if f.cs.SortKeys {
   312  				sortValues(keys, f.cs)
   313  			}
   314  			for i, key := range keys {
   315  				if i > 0 {
   316  					f.fs.Write(spaceBytes)
   317  				}
   318  				f.ignoreNextType = true
   319  				f.format(f.unpackValue(key))
   320  				f.fs.Write(colonBytes)
   321  				f.ignoreNextType = true
   322  				f.format(f.unpackValue(v.MapIndex(key)))
   323  			}
   324  		}
   325  		f.depth--
   326  		f.fs.Write(closeMapBytes)
   327  
   328  	case reflect.Struct:
   329  		numFields := v.NumField()
   330  		f.fs.Write(openBraceBytes)
   331  		f.depth++
   332  		if (f.cs.MaxDepth != 0) && (f.depth > f.cs.MaxDepth) {
   333  			f.fs.Write(maxShortBytes)
   334  		} else {
   335  			vt := v.Type()
   336  			for i := 0; i < numFields; i++ {
   337  				if i > 0 {
   338  					f.fs.Write(spaceBytes)
   339  				}
   340  				vtf := vt.Field(i)
   341  				if f.fs.Flag('+') || f.fs.Flag('#') {
   342  					f.fs.Write([]byte(vtf.Name))
   343  					f.fs.Write(colonBytes)
   344  				}
   345  				f.format(f.unpackValue(v.Field(i)))
   346  			}
   347  		}
   348  		f.depth--
   349  		f.fs.Write(closeBraceBytes)
   350  
   351  	case reflect.Uintptr:
   352  		printHexPtr(f.fs, uintptr(v.Uint()))
   353  
   354  	case reflect.UnsafePointer, reflect.Chan, reflect.Func:
   355  		printHexPtr(f.fs, v.Pointer())
   356  
   357  	// There were not any other types at the time this code was written, but
   358  	// fall back to letting the default fmt package handle it if any get added.
   359  	default:
   360  		format := f.buildDefaultFormat()
   361  		if v.CanInterface() {
   362  			fmt.Fprintf(f.fs, format, v.Interface())
   363  		} else {
   364  			fmt.Fprintf(f.fs, format, v.String())
   365  		}
   366  	}
   367  }
   368  
   369  // Format satisfies the fmt.Formatter interface. See NewFormatter for usage
   370  // details.
   371  func (f *formatState) Format(fs fmt.State, verb rune) {
   372  	f.fs = fs
   373  
   374  	// Use standard formatting for verbs that are not v.
   375  	if verb != 'v' {
   376  		format := f.constructOrigFormat(verb)
   377  		fmt.Fprintf(fs, format, f.value)
   378  		return
   379  	}
   380  
   381  	if f.value == nil {
   382  		if fs.Flag('#') {
   383  			fs.Write(interfaceBytes)
   384  		}
   385  		fs.Write(nilAngleBytes)
   386  		return
   387  	}
   388  
   389  	f.format(reflect.ValueOf(f.value))
   390  }
   391  
   392  // newFormatter is a helper function to consolidate the logic from the various
   393  // public methods which take varying config states.
   394  func newFormatter(cs *ConfigState, v interface{}) fmt.Formatter {
   395  	fs := &formatState{value: v, cs: cs}
   396  	fs.pointers = make(map[uintptr]int)
   397  	return fs
   398  }
   399  
   400  /*
   401  NewFormatter returns a custom formatter that satisfies the fmt.Formatter
   402  interface.  As a result, it integrates cleanly with standard fmt package
   403  printing functions.  The formatter is useful for inline printing of smaller data
   404  types similar to the standard %v format specifier.
   405  
   406  The custom formatter only responds to the %v (most compact), %+v (adds pointer
   407  addresses), %#v (adds types), or %#+v (adds types and pointer addresses) verb
   408  combinations.  Any other verbs such as %x and %q will be sent to the the
   409  standard fmt package for formatting.  In addition, the custom formatter ignores
   410  the width and precision arguments (however they will still work on the format
   411  specifiers not handled by the custom formatter).
   412  
   413  Typically this function shouldn't be called directly.  It is much easier to make
   414  use of the custom formatter by calling one of the convenience functions such as
   415  Printf, Println, or Fprintf.
   416  */
   417  func NewFormatter(v interface{}) fmt.Formatter {
   418  	return newFormatter(&Config, v)
   419  }