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.

202 lines
8.0 KiB

  1. /*
  2. * Copyright (c) 2013 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 implements a deep pretty printer for Go data structures to aid in
  18. debugging.
  19. A quick overview of the additional features spew provides over the built-in
  20. printing facilities for Go data types are as follows:
  21. * Pointers are dereferenced and followed
  22. * Circular data structures are detected and handled properly
  23. * Custom Stringer/error interfaces are optionally invoked, including
  24. on unexported types
  25. * Custom types which only implement the Stringer/error interfaces via
  26. a pointer receiver are optionally invoked when passing non-pointer
  27. variables
  28. * Byte arrays and slices are dumped like the hexdump -C command which
  29. includes offsets, byte values in hex, and ASCII output (only when using
  30. Dump style)
  31. There are two different approaches spew allows for dumping Go data structures:
  32. * Dump style which prints with newlines, customizable indentation,
  33. and additional debug information such as types and all pointer addresses
  34. used to indirect to the final value
  35. * A custom Formatter interface that integrates cleanly with the standard fmt
  36. package and replaces %v, %+v, %#v, and %#+v to provide inline printing
  37. similar to the default %v while providing the additional functionality
  38. outlined above and passing unsupported format verbs such as %x and %q
  39. along to fmt
  40. Quick Start
  41. This section demonstrates how to quickly get started with spew. See the
  42. sections below for further details on formatting and configuration options.
  43. To dump a variable with full newlines, indentation, type, and pointer
  44. information use Dump, Fdump, or Sdump:
  45. spew.Dump(myVar1, myVar2, ...)
  46. spew.Fdump(someWriter, myVar1, myVar2, ...)
  47. str := spew.Sdump(myVar1, myVar2, ...)
  48. Alternatively, if you would prefer to use format strings with a compacted inline
  49. printing style, use the convenience wrappers Printf, Fprintf, etc with
  50. %v (most compact), %+v (adds pointer addresses), %#v (adds types), or
  51. %#+v (adds types and pointer addresses):
  52. spew.Printf("myVar1: %v -- myVar2: %+v", myVar1, myVar2)
  53. spew.Printf("myVar3: %#v -- myVar4: %#+v", myVar3, myVar4)
  54. spew.Fprintf(someWriter, "myVar1: %v -- myVar2: %+v", myVar1, myVar2)
  55. spew.Fprintf(someWriter, "myVar3: %#v -- myVar4: %#+v", myVar3, myVar4)
  56. Configuration Options
  57. Configuration of spew is handled by fields in the ConfigState type. For
  58. convenience, all of the top-level functions use a global state available
  59. via the spew.Config global.
  60. It is also possible to create a ConfigState instance that provides methods
  61. equivalent to the top-level functions. This allows concurrent configuration
  62. options. See the ConfigState documentation for more details.
  63. The following configuration options are available:
  64. * Indent
  65. String to use for each indentation level for Dump functions.
  66. It is a single space by default. A popular alternative is "\t".
  67. * MaxDepth
  68. Maximum number of levels to descend into nested data structures.
  69. There is no limit by default.
  70. * DisableMethods
  71. Disables invocation of error and Stringer interface methods.
  72. Method invocation is enabled by default.
  73. * DisablePointerMethods
  74. Disables invocation of error and Stringer interface methods on types
  75. which only accept pointer receivers from non-pointer variables.
  76. Pointer method invocation is enabled by default.
  77. * ContinueOnMethod
  78. Enables recursion into types after invoking error and Stringer interface
  79. methods. Recursion after method invocation is disabled by default.
  80. * SortKeys
  81. Specifies map keys should be sorted before being printed. Use
  82. this to have a more deterministic, diffable output. Note that
  83. only native types (bool, int, uint, floats, uintptr and string)
  84. and types which implement error or Stringer interfaces are
  85. supported with other types sorted according to the
  86. reflect.Value.String() output which guarantees display
  87. stability. Natural map order is used by default.
  88. * SpewKeys
  89. Specifies that, as a last resort attempt, map keys should be
  90. spewed to strings and sorted by those strings. This is only
  91. considered if SortKeys is true.
  92. Dump Usage
  93. Simply call spew.Dump with a list of variables you want to dump:
  94. spew.Dump(myVar1, myVar2, ...)
  95. You may also call spew.Fdump if you would prefer to output to an arbitrary
  96. io.Writer. For example, to dump to standard error:
  97. spew.Fdump(os.Stderr, myVar1, myVar2, ...)
  98. A third option is to call spew.Sdump to get the formatted output as a string:
  99. str := spew.Sdump(myVar1, myVar2, ...)
  100. Sample Dump Output
  101. See the Dump example for details on the setup of the types and variables being
  102. shown here.
  103. (main.Foo) {
  104. unexportedField: (*main.Bar)(0xf84002e210)({
  105. flag: (main.Flag) flagTwo,
  106. data: (uintptr) <nil>
  107. }),
  108. ExportedField: (map[interface {}]interface {}) (len=1) {
  109. (string) (len=3) "one": (bool) true
  110. }
  111. }
  112. Byte (and uint8) arrays and slices are displayed uniquely like the hexdump -C
  113. command as shown.
  114. ([]uint8) (len=32 cap=32) {
  115. 00000000 11 12 13 14 15 16 17 18 19 1a 1b 1c 1d 1e 1f 20 |............... |
  116. 00000010 21 22 23 24 25 26 27 28 29 2a 2b 2c 2d 2e 2f 30 |!"#$%&'()*+,-./0|
  117. 00000020 31 32 |12|
  118. }
  119. Custom Formatter
  120. Spew provides a custom formatter that implements the fmt.Formatter interface
  121. so that it integrates cleanly with standard fmt package printing functions. The
  122. formatter is useful for inline printing of smaller data types similar to the
  123. standard %v format specifier.
  124. The custom formatter only responds to the %v (most compact), %+v (adds pointer
  125. addresses), %#v (adds types), or %#+v (adds types and pointer addresses) verb
  126. combinations. Any other verbs such as %x and %q will be sent to the the
  127. standard fmt package for formatting. In addition, the custom formatter ignores
  128. the width and precision arguments (however they will still work on the format
  129. specifiers not handled by the custom formatter).
  130. Custom Formatter Usage
  131. The simplest way to make use of the spew custom formatter is to call one of the
  132. convenience functions such as spew.Printf, spew.Println, or spew.Printf. The
  133. functions have syntax you are most likely already familiar with:
  134. spew.Printf("myVar1: %v -- myVar2: %+v", myVar1, myVar2)
  135. spew.Printf("myVar3: %#v -- myVar4: %#+v", myVar3, myVar4)
  136. spew.Println(myVar, myVar2)
  137. spew.Fprintf(os.Stderr, "myVar1: %v -- myVar2: %+v", myVar1, myVar2)
  138. spew.Fprintf(os.Stderr, "myVar3: %#v -- myVar4: %#+v", myVar3, myVar4)
  139. See the Index for the full list convenience functions.
  140. Sample Formatter Output
  141. Double pointer to a uint8:
  142. %v: <**>5
  143. %+v: <**>(0xf8400420d0->0xf8400420c8)5
  144. %#v: (**uint8)5
  145. %#+v: (**uint8)(0xf8400420d0->0xf8400420c8)5
  146. Pointer to circular struct with a uint8 field and a pointer to itself:
  147. %v: <*>{1 <*><shown>}
  148. %+v: <*>(0xf84003e260){ui8:1 c:<*>(0xf84003e260)<shown>}
  149. %#v: (*main.circular){ui8:(uint8)1 c:(*main.circular)<shown>}
  150. %#+v: (*main.circular)(0xf84003e260){ui8:(uint8)1 c:(*main.circular)(0xf84003e260)<shown>}
  151. See the Printf example for details on the setup of variables being shown
  152. here.
  153. Errors
  154. Since it is possible for custom Stringer/error interfaces to panic, spew
  155. detects them and handles them internally by printing the panic information
  156. inline with the output. Since spew is intended to provide deep pretty printing
  157. capabilities on structures, it intentionally does not return any errors.
  158. */
  159. package spew