gostring.go 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386
  1. // Protocol Buffers for Go with Gadgets
  2. //
  3. // Copyright (c) 2013, The GoGo Authors. All rights reserved.
  4. // http://github.com/gogo/protobuf
  5. //
  6. // Redistribution and use in source and binary forms, with or without
  7. // modification, are permitted provided that the following conditions are
  8. // met:
  9. //
  10. // * Redistributions of source code must retain the above copyright
  11. // notice, this list of conditions and the following disclaimer.
  12. // * Redistributions in binary form must reproduce the above
  13. // copyright notice, this list of conditions and the following disclaimer
  14. // in the documentation and/or other materials provided with the
  15. // distribution.
  16. //
  17. // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
  18. // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
  19. // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
  20. // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
  21. // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
  22. // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
  23. // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
  24. // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
  25. // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
  26. // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
  27. // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  28. /*
  29. The gostring plugin generates a GoString method for each message.
  30. The GoString method is called whenever you use a fmt.Printf as such:
  31. fmt.Printf("%#v", mymessage)
  32. or whenever you actually call GoString()
  33. The output produced by the GoString method can be copied from the output into code and used to set a variable.
  34. It is totally valid Go Code and is populated exactly as the struct that was printed out.
  35. It is enabled by the following extensions:
  36. - gostring
  37. - gostring_all
  38. The gostring plugin also generates a test given it is enabled using one of the following extensions:
  39. - testgen
  40. - testgen_all
  41. Let us look at:
  42. github.com/gogo/protobuf/test/example/example.proto
  43. Btw all the output can be seen at:
  44. github.com/gogo/protobuf/test/example/*
  45. The following message:
  46. option (gogoproto.gostring_all) = true;
  47. message A {
  48. optional string Description = 1 [(gogoproto.nullable) = false];
  49. optional int64 Number = 2 [(gogoproto.nullable) = false];
  50. optional bytes Id = 3 [(gogoproto.customtype) = "github.com/gogo/protobuf/test/custom.Uuid", (gogoproto.nullable) = false];
  51. }
  52. given to the gostring plugin, will generate the following code:
  53. func (this *A) GoString() string {
  54. if this == nil {
  55. return "nil"
  56. }
  57. s := strings1.Join([]string{`&test.A{` + `Description:` + fmt1.Sprintf("%#v", this.Description), `Number:` + fmt1.Sprintf("%#v", this.Number), `Id:` + fmt1.Sprintf("%#v", this.Id), `XXX_unrecognized:` + fmt1.Sprintf("%#v", this.XXX_unrecognized) + `}`}, ", ")
  58. return s
  59. }
  60. and the following test code:
  61. func TestAGoString(t *testing6.T) {
  62. popr := math_rand6.New(math_rand6.NewSource(time6.Now().UnixNano()))
  63. p := NewPopulatedA(popr, false)
  64. s1 := p.GoString()
  65. s2 := fmt2.Sprintf("%#v", p)
  66. if s1 != s2 {
  67. t.Fatalf("GoString want %v got %v", s1, s2)
  68. }
  69. _, err := go_parser.ParseExpr(s1)
  70. if err != nil {
  71. panic(err)
  72. }
  73. }
  74. Typically fmt.Printf("%#v") will stop to print when it reaches a pointer and
  75. not print their values, while the generated GoString method will always print all values, recursively.
  76. */
  77. package gostring
  78. import (
  79. "fmt"
  80. "os"
  81. "strconv"
  82. "strings"
  83. "github.com/gogo/protobuf/gogoproto"
  84. "github.com/gogo/protobuf/protoc-gen-gogo/generator"
  85. )
  86. type gostring struct {
  87. *generator.Generator
  88. generator.PluginImports
  89. atleastOne bool
  90. localName string
  91. overwrite bool
  92. }
  93. func NewGoString() *gostring {
  94. return &gostring{}
  95. }
  96. func (p *gostring) Name() string {
  97. return "gostring"
  98. }
  99. func (p *gostring) Overwrite() {
  100. p.overwrite = true
  101. }
  102. func (p *gostring) Init(g *generator.Generator) {
  103. p.Generator = g
  104. }
  105. func (p *gostring) Generate(file *generator.FileDescriptor) {
  106. proto3 := gogoproto.IsProto3(file.FileDescriptorProto)
  107. p.PluginImports = generator.NewPluginImports(p.Generator)
  108. p.atleastOne = false
  109. p.localName = generator.FileName(file)
  110. fmtPkg := p.NewImport("fmt")
  111. stringsPkg := p.NewImport("strings")
  112. protoPkg := p.NewImport("github.com/gogo/protobuf/proto")
  113. if !gogoproto.ImportsGoGoProto(file.FileDescriptorProto) {
  114. protoPkg = p.NewImport("github.com/golang/protobuf/proto")
  115. }
  116. sortPkg := p.NewImport("sort")
  117. strconvPkg := p.NewImport("strconv")
  118. reflectPkg := p.NewImport("reflect")
  119. sortKeysPkg := p.NewImport("github.com/gogo/protobuf/sortkeys")
  120. extensionToGoStringUsed := false
  121. for _, message := range file.Messages() {
  122. if !p.overwrite && !gogoproto.HasGoString(file.FileDescriptorProto, message.DescriptorProto) {
  123. continue
  124. }
  125. if message.DescriptorProto.GetOptions().GetMapEntry() {
  126. continue
  127. }
  128. p.atleastOne = true
  129. packageName := file.GoPackageName()
  130. ccTypeName := generator.CamelCaseSlice(message.TypeName())
  131. p.P(`func (this *`, ccTypeName, `) GoString() string {`)
  132. p.In()
  133. p.P(`if this == nil {`)
  134. p.In()
  135. p.P(`return "nil"`)
  136. p.Out()
  137. p.P(`}`)
  138. p.P(`s := make([]string, 0, `, strconv.Itoa(len(message.Field)+4), `)`)
  139. p.P(`s = append(s, "&`, packageName, ".", ccTypeName, `{")`)
  140. oneofs := make(map[string]struct{})
  141. for _, field := range message.Field {
  142. nullable := gogoproto.IsNullable(field)
  143. repeated := field.IsRepeated()
  144. fieldname := p.GetFieldName(message, field)
  145. oneof := field.OneofIndex != nil
  146. if oneof {
  147. if _, ok := oneofs[fieldname]; ok {
  148. continue
  149. } else {
  150. oneofs[fieldname] = struct{}{}
  151. }
  152. p.P(`if this.`, fieldname, ` != nil {`)
  153. p.In()
  154. p.P(`s = append(s, "`, fieldname, `: " + `, fmtPkg.Use(), `.Sprintf("%#v", this.`, fieldname, `) + ",\n")`)
  155. p.Out()
  156. p.P(`}`)
  157. } else if p.IsMap(field) {
  158. m := p.GoMapType(nil, field)
  159. mapgoTyp, keyField, keyAliasField := m.GoType, m.KeyField, m.KeyAliasField
  160. keysName := `keysFor` + fieldname
  161. keygoTyp, _ := p.GoType(nil, keyField)
  162. keygoTyp = strings.Replace(keygoTyp, "*", "", 1)
  163. keygoAliasTyp, _ := p.GoType(nil, keyAliasField)
  164. keygoAliasTyp = strings.Replace(keygoAliasTyp, "*", "", 1)
  165. keyCapTyp := generator.CamelCase(keygoTyp)
  166. p.P(keysName, ` := make([]`, keygoTyp, `, 0, len(this.`, fieldname, `))`)
  167. p.P(`for k, _ := range this.`, fieldname, ` {`)
  168. p.In()
  169. if keygoAliasTyp == keygoTyp {
  170. p.P(keysName, ` = append(`, keysName, `, k)`)
  171. } else {
  172. p.P(keysName, ` = append(`, keysName, `, `, keygoTyp, `(k))`)
  173. }
  174. p.Out()
  175. p.P(`}`)
  176. p.P(sortKeysPkg.Use(), `.`, keyCapTyp, `s(`, keysName, `)`)
  177. mapName := `mapStringFor` + fieldname
  178. p.P(mapName, ` := "`, mapgoTyp, `{"`)
  179. p.P(`for _, k := range `, keysName, ` {`)
  180. p.In()
  181. if keygoAliasTyp == keygoTyp {
  182. p.P(mapName, ` += fmt.Sprintf("%#v: %#v,", k, this.`, fieldname, `[k])`)
  183. } else {
  184. p.P(mapName, ` += fmt.Sprintf("%#v: %#v,", k, this.`, fieldname, `[`, keygoAliasTyp, `(k)])`)
  185. }
  186. p.Out()
  187. p.P(`}`)
  188. p.P(mapName, ` += "}"`)
  189. p.P(`if this.`, fieldname, ` != nil {`)
  190. p.In()
  191. p.P(`s = append(s, "`, fieldname, `: " + `, mapName, `+ ",\n")`)
  192. p.Out()
  193. p.P(`}`)
  194. } else if (field.IsMessage() && !gogoproto.IsCustomType(field) && !gogoproto.IsStdTime(field) && !gogoproto.IsStdDuration(field)) || p.IsGroup(field) {
  195. if nullable || repeated {
  196. p.P(`if this.`, fieldname, ` != nil {`)
  197. p.In()
  198. }
  199. if nullable {
  200. p.P(`s = append(s, "`, fieldname, `: " + `, fmtPkg.Use(), `.Sprintf("%#v", this.`, fieldname, `) + ",\n")`)
  201. } else if repeated {
  202. if nullable {
  203. p.P(`s = append(s, "`, fieldname, `: " + `, fmtPkg.Use(), `.Sprintf("%#v", this.`, fieldname, `) + ",\n")`)
  204. } else {
  205. goTyp, _ := p.GoType(message, field)
  206. goTyp = strings.Replace(goTyp, "[]", "", 1)
  207. p.P("vs := make([]*", goTyp, ", len(this.", fieldname, "))")
  208. p.P("for i := range vs {")
  209. p.In()
  210. p.P("vs[i] = &this.", fieldname, "[i]")
  211. p.Out()
  212. p.P("}")
  213. p.P(`s = append(s, "`, fieldname, `: " + `, fmtPkg.Use(), `.Sprintf("%#v", vs) + ",\n")`)
  214. }
  215. } else {
  216. p.P(`s = append(s, "`, fieldname, `: " + `, stringsPkg.Use(), `.Replace(this.`, fieldname, `.GoString()`, ",`&`,``,1)", ` + ",\n")`)
  217. }
  218. if nullable || repeated {
  219. p.Out()
  220. p.P(`}`)
  221. }
  222. } else {
  223. if !proto3 && (nullable || repeated) {
  224. p.P(`if this.`, fieldname, ` != nil {`)
  225. p.In()
  226. }
  227. if field.IsEnum() {
  228. if nullable && !repeated && !proto3 {
  229. goTyp, _ := p.GoType(message, field)
  230. p.P(`s = append(s, "`, fieldname, `: " + valueToGoString`, p.localName, `(this.`, fieldname, `,"`, generator.GoTypeToName(goTyp), `"`, `) + ",\n")`)
  231. } else {
  232. p.P(`s = append(s, "`, fieldname, `: " + `, fmtPkg.Use(), `.Sprintf("%#v", this.`, fieldname, `) + ",\n")`)
  233. }
  234. } else {
  235. if nullable && !repeated && !proto3 {
  236. goTyp, _ := p.GoType(message, field)
  237. p.P(`s = append(s, "`, fieldname, `: " + valueToGoString`, p.localName, `(this.`, fieldname, `,"`, generator.GoTypeToName(goTyp), `"`, `) + ",\n")`)
  238. } else {
  239. p.P(`s = append(s, "`, fieldname, `: " + `, fmtPkg.Use(), `.Sprintf("%#v", this.`, fieldname, `) + ",\n")`)
  240. }
  241. }
  242. if !proto3 && (nullable || repeated) {
  243. p.Out()
  244. p.P(`}`)
  245. }
  246. }
  247. }
  248. if message.DescriptorProto.HasExtension() {
  249. if gogoproto.HasExtensionsMap(file.FileDescriptorProto, message.DescriptorProto) {
  250. p.P(`s = append(s, "XXX_InternalExtensions: " + extensionToGoString`, p.localName, `(this) + ",\n")`)
  251. extensionToGoStringUsed = true
  252. } else {
  253. p.P(`if this.XXX_extensions != nil {`)
  254. p.In()
  255. p.P(`s = append(s, "XXX_extensions: " + `, fmtPkg.Use(), `.Sprintf("%#v", this.XXX_extensions) + ",\n")`)
  256. p.Out()
  257. p.P(`}`)
  258. }
  259. }
  260. if gogoproto.HasUnrecognized(file.FileDescriptorProto, message.DescriptorProto) {
  261. p.P(`if this.XXX_unrecognized != nil {`)
  262. p.In()
  263. p.P(`s = append(s, "XXX_unrecognized:" + `, fmtPkg.Use(), `.Sprintf("%#v", this.XXX_unrecognized) + ",\n")`)
  264. p.Out()
  265. p.P(`}`)
  266. }
  267. p.P(`s = append(s, "}")`)
  268. p.P(`return `, stringsPkg.Use(), `.Join(s, "")`)
  269. p.Out()
  270. p.P(`}`)
  271. //Generate GoString methods for oneof fields
  272. for _, field := range message.Field {
  273. oneof := field.OneofIndex != nil
  274. if !oneof {
  275. continue
  276. }
  277. ccTypeName := p.OneOfTypeName(message, field)
  278. p.P(`func (this *`, ccTypeName, `) GoString() string {`)
  279. p.In()
  280. p.P(`if this == nil {`)
  281. p.In()
  282. p.P(`return "nil"`)
  283. p.Out()
  284. p.P(`}`)
  285. fieldname := p.GetOneOfFieldName(message, field)
  286. outStr := strings.Join([]string{
  287. "s := ",
  288. stringsPkg.Use(), ".Join([]string{`&", packageName, ".", ccTypeName, "{` + \n",
  289. "`", fieldname, ":` + ", fmtPkg.Use(), `.Sprintf("%#v", this.`, fieldname, `)`,
  290. " + `}`",
  291. `}`,
  292. `,", "`,
  293. `)`}, "")
  294. p.P(outStr)
  295. p.P(`return s`)
  296. p.Out()
  297. p.P(`}`)
  298. }
  299. }
  300. if !p.atleastOne {
  301. return
  302. }
  303. p.P(`func valueToGoString`, p.localName, `(v interface{}, typ string) string {`)
  304. p.In()
  305. p.P(`rv := `, reflectPkg.Use(), `.ValueOf(v)`)
  306. p.P(`if rv.IsNil() {`)
  307. p.In()
  308. p.P(`return "nil"`)
  309. p.Out()
  310. p.P(`}`)
  311. p.P(`pv := `, reflectPkg.Use(), `.Indirect(rv).Interface()`)
  312. p.P(`return `, fmtPkg.Use(), `.Sprintf("func(v %v) *%v { return &v } ( %#v )", typ, typ, pv)`)
  313. p.Out()
  314. p.P(`}`)
  315. if extensionToGoStringUsed {
  316. if !gogoproto.ImportsGoGoProto(file.FileDescriptorProto) {
  317. fmt.Fprintf(os.Stderr, "The GoString plugin for messages with extensions requires importing gogoprotobuf. Please see file %s", file.GetName())
  318. os.Exit(1)
  319. }
  320. p.P(`func extensionToGoString`, p.localName, `(m `, protoPkg.Use(), `.Message) string {`)
  321. p.In()
  322. p.P(`e := `, protoPkg.Use(), `.GetUnsafeExtensionsMap(m)`)
  323. p.P(`if e == nil { return "nil" }`)
  324. p.P(`s := "proto.NewUnsafeXXX_InternalExtensions(map[int32]proto.Extension{"`)
  325. p.P(`keys := make([]int, 0, len(e))`)
  326. p.P(`for k := range e {`)
  327. p.In()
  328. p.P(`keys = append(keys, int(k))`)
  329. p.Out()
  330. p.P(`}`)
  331. p.P(sortPkg.Use(), `.Ints(keys)`)
  332. p.P(`ss := []string{}`)
  333. p.P(`for _, k := range keys {`)
  334. p.In()
  335. p.P(`ss = append(ss, `, strconvPkg.Use(), `.Itoa(k) + ": " + e[int32(k)].GoString())`)
  336. p.Out()
  337. p.P(`}`)
  338. p.P(`s+=`, stringsPkg.Use(), `.Join(ss, ",") + "})"`)
  339. p.P(`return s`)
  340. p.Out()
  341. p.P(`}`)
  342. }
  343. }
  344. func init() {
  345. generator.RegisterPlugin(NewGoString())
  346. }