format.go 7.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277
  1. // Copyright 2017, The Go Authors. All rights reserved.
  2. // Use of this source code is governed by a BSD-style
  3. // license that can be found in the LICENSE.md file.
  4. // Package value provides functionality for reflect.Value types.
  5. package value
  6. import (
  7. "fmt"
  8. "reflect"
  9. "strconv"
  10. "strings"
  11. "unicode"
  12. )
  13. var stringerIface = reflect.TypeOf((*fmt.Stringer)(nil)).Elem()
  14. // Format formats the value v as a string.
  15. //
  16. // This is similar to fmt.Sprintf("%+v", v) except this:
  17. // * Prints the type unless it can be elided
  18. // * Avoids printing struct fields that are zero
  19. // * Prints a nil-slice as being nil, not empty
  20. // * Prints map entries in deterministic order
  21. func Format(v reflect.Value, conf FormatConfig) string {
  22. conf.printType = true
  23. conf.followPointers = true
  24. conf.realPointers = true
  25. return formatAny(v, conf, nil)
  26. }
  27. type FormatConfig struct {
  28. UseStringer bool // Should the String method be used if available?
  29. printType bool // Should we print the type before the value?
  30. PrintPrimitiveType bool // Should we print the type of primitives?
  31. followPointers bool // Should we recursively follow pointers?
  32. realPointers bool // Should we print the real address of pointers?
  33. }
  34. func formatAny(v reflect.Value, conf FormatConfig, visited map[uintptr]bool) string {
  35. // TODO: Should this be a multi-line printout in certain situations?
  36. if !v.IsValid() {
  37. return "<non-existent>"
  38. }
  39. if conf.UseStringer && v.Type().Implements(stringerIface) && v.CanInterface() {
  40. if (v.Kind() == reflect.Ptr || v.Kind() == reflect.Interface) && v.IsNil() {
  41. return "<nil>"
  42. }
  43. const stringerPrefix = "s" // Indicates that the String method was used
  44. s := v.Interface().(fmt.Stringer).String()
  45. return stringerPrefix + formatString(s)
  46. }
  47. switch v.Kind() {
  48. case reflect.Bool:
  49. return formatPrimitive(v.Type(), v.Bool(), conf)
  50. case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:
  51. return formatPrimitive(v.Type(), v.Int(), conf)
  52. case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Uintptr:
  53. if v.Type().PkgPath() == "" || v.Kind() == reflect.Uintptr {
  54. // Unnamed uints are usually bytes or words, so use hexadecimal.
  55. return formatPrimitive(v.Type(), formatHex(v.Uint()), conf)
  56. }
  57. return formatPrimitive(v.Type(), v.Uint(), conf)
  58. case reflect.Float32, reflect.Float64:
  59. return formatPrimitive(v.Type(), v.Float(), conf)
  60. case reflect.Complex64, reflect.Complex128:
  61. return formatPrimitive(v.Type(), v.Complex(), conf)
  62. case reflect.String:
  63. return formatPrimitive(v.Type(), formatString(v.String()), conf)
  64. case reflect.UnsafePointer, reflect.Chan, reflect.Func:
  65. return formatPointer(v, conf)
  66. case reflect.Ptr:
  67. if v.IsNil() {
  68. if conf.printType {
  69. return fmt.Sprintf("(%v)(nil)", v.Type())
  70. }
  71. return "<nil>"
  72. }
  73. if visited[v.Pointer()] || !conf.followPointers {
  74. return formatPointer(v, conf)
  75. }
  76. visited = insertPointer(visited, v.Pointer())
  77. return "&" + formatAny(v.Elem(), conf, visited)
  78. case reflect.Interface:
  79. if v.IsNil() {
  80. if conf.printType {
  81. return fmt.Sprintf("%v(nil)", v.Type())
  82. }
  83. return "<nil>"
  84. }
  85. return formatAny(v.Elem(), conf, visited)
  86. case reflect.Slice:
  87. if v.IsNil() {
  88. if conf.printType {
  89. return fmt.Sprintf("%v(nil)", v.Type())
  90. }
  91. return "<nil>"
  92. }
  93. if visited[v.Pointer()] {
  94. return formatPointer(v, conf)
  95. }
  96. visited = insertPointer(visited, v.Pointer())
  97. fallthrough
  98. case reflect.Array:
  99. var ss []string
  100. subConf := conf
  101. subConf.printType = v.Type().Elem().Kind() == reflect.Interface
  102. for i := 0; i < v.Len(); i++ {
  103. s := formatAny(v.Index(i), subConf, visited)
  104. ss = append(ss, s)
  105. }
  106. s := fmt.Sprintf("{%s}", strings.Join(ss, ", "))
  107. if conf.printType {
  108. return v.Type().String() + s
  109. }
  110. return s
  111. case reflect.Map:
  112. if v.IsNil() {
  113. if conf.printType {
  114. return fmt.Sprintf("%v(nil)", v.Type())
  115. }
  116. return "<nil>"
  117. }
  118. if visited[v.Pointer()] {
  119. return formatPointer(v, conf)
  120. }
  121. visited = insertPointer(visited, v.Pointer())
  122. var ss []string
  123. keyConf, valConf := conf, conf
  124. keyConf.printType = v.Type().Key().Kind() == reflect.Interface
  125. keyConf.followPointers = false
  126. valConf.printType = v.Type().Elem().Kind() == reflect.Interface
  127. for _, k := range SortKeys(v.MapKeys()) {
  128. sk := formatAny(k, keyConf, visited)
  129. sv := formatAny(v.MapIndex(k), valConf, visited)
  130. ss = append(ss, fmt.Sprintf("%s: %s", sk, sv))
  131. }
  132. s := fmt.Sprintf("{%s}", strings.Join(ss, ", "))
  133. if conf.printType {
  134. return v.Type().String() + s
  135. }
  136. return s
  137. case reflect.Struct:
  138. var ss []string
  139. subConf := conf
  140. subConf.printType = true
  141. for i := 0; i < v.NumField(); i++ {
  142. vv := v.Field(i)
  143. if isZero(vv) {
  144. continue // Elide zero value fields
  145. }
  146. name := v.Type().Field(i).Name
  147. subConf.UseStringer = conf.UseStringer
  148. s := formatAny(vv, subConf, visited)
  149. ss = append(ss, fmt.Sprintf("%s: %s", name, s))
  150. }
  151. s := fmt.Sprintf("{%s}", strings.Join(ss, ", "))
  152. if conf.printType {
  153. return v.Type().String() + s
  154. }
  155. return s
  156. default:
  157. panic(fmt.Sprintf("%v kind not handled", v.Kind()))
  158. }
  159. }
  160. func formatString(s string) string {
  161. // Use quoted string if it the same length as a raw string literal.
  162. // Otherwise, attempt to use the raw string form.
  163. qs := strconv.Quote(s)
  164. if len(qs) == 1+len(s)+1 {
  165. return qs
  166. }
  167. // Disallow newlines to ensure output is a single line.
  168. // Only allow printable runes for readability purposes.
  169. rawInvalid := func(r rune) bool {
  170. return r == '`' || r == '\n' || !unicode.IsPrint(r)
  171. }
  172. if strings.IndexFunc(s, rawInvalid) < 0 {
  173. return "`" + s + "`"
  174. }
  175. return qs
  176. }
  177. func formatPrimitive(t reflect.Type, v interface{}, conf FormatConfig) string {
  178. if conf.printType && (conf.PrintPrimitiveType || t.PkgPath() != "") {
  179. return fmt.Sprintf("%v(%v)", t, v)
  180. }
  181. return fmt.Sprintf("%v", v)
  182. }
  183. func formatPointer(v reflect.Value, conf FormatConfig) string {
  184. p := v.Pointer()
  185. if !conf.realPointers {
  186. p = 0 // For deterministic printing purposes
  187. }
  188. s := formatHex(uint64(p))
  189. if conf.printType {
  190. return fmt.Sprintf("(%v)(%s)", v.Type(), s)
  191. }
  192. return s
  193. }
  194. func formatHex(u uint64) string {
  195. var f string
  196. switch {
  197. case u <= 0xff:
  198. f = "0x%02x"
  199. case u <= 0xffff:
  200. f = "0x%04x"
  201. case u <= 0xffffff:
  202. f = "0x%06x"
  203. case u <= 0xffffffff:
  204. f = "0x%08x"
  205. case u <= 0xffffffffff:
  206. f = "0x%010x"
  207. case u <= 0xffffffffffff:
  208. f = "0x%012x"
  209. case u <= 0xffffffffffffff:
  210. f = "0x%014x"
  211. case u <= 0xffffffffffffffff:
  212. f = "0x%016x"
  213. }
  214. return fmt.Sprintf(f, u)
  215. }
  216. // insertPointer insert p into m, allocating m if necessary.
  217. func insertPointer(m map[uintptr]bool, p uintptr) map[uintptr]bool {
  218. if m == nil {
  219. m = make(map[uintptr]bool)
  220. }
  221. m[p] = true
  222. return m
  223. }
  224. // isZero reports whether v is the zero value.
  225. // This does not rely on Interface and so can be used on unexported fields.
  226. func isZero(v reflect.Value) bool {
  227. switch v.Kind() {
  228. case reflect.Bool:
  229. return v.Bool() == false
  230. case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:
  231. return v.Int() == 0
  232. case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Uintptr:
  233. return v.Uint() == 0
  234. case reflect.Float32, reflect.Float64:
  235. return v.Float() == 0
  236. case reflect.Complex64, reflect.Complex128:
  237. return v.Complex() == 0
  238. case reflect.String:
  239. return v.String() == ""
  240. case reflect.UnsafePointer:
  241. return v.Pointer() == 0
  242. case reflect.Chan, reflect.Func, reflect.Interface, reflect.Ptr, reflect.Map, reflect.Slice:
  243. return v.IsNil()
  244. case reflect.Array:
  245. for i := 0; i < v.Len(); i++ {
  246. if !isZero(v.Index(i)) {
  247. return false
  248. }
  249. }
  250. return true
  251. case reflect.Struct:
  252. for i := 0; i < v.NumField(); i++ {
  253. if !isZero(v.Field(i)) {
  254. return false
  255. }
  256. }
  257. return true
  258. }
  259. return false
  260. }