face.go 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233
  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 face plugin generates a function will be generated which can convert a structure which satisfies an interface (face) to the specified structure.
  30. This interface contains getters for each of the fields in the struct.
  31. The specified struct is also generated with the getters.
  32. This means that getters should be turned off so as not to conflict with face getters.
  33. This allows it to satisfy its own face.
  34. It is enabled by the following extensions:
  35. - face
  36. - face_all
  37. Turn off getters by using the following extensions:
  38. - getters
  39. - getters_all
  40. The face plugin also generates a test given it is enabled using one of the following extensions:
  41. - testgen
  42. - testgen_all
  43. Let us look at:
  44. github.com/gogo/protobuf/test/example/example.proto
  45. Btw all the output can be seen at:
  46. github.com/gogo/protobuf/test/example/*
  47. The following message:
  48. message A {
  49. option (gogoproto.face) = true;
  50. option (gogoproto.goproto_getters) = false;
  51. optional string Description = 1 [(gogoproto.nullable) = false];
  52. optional int64 Number = 2 [(gogoproto.nullable) = false];
  53. optional bytes Id = 3 [(gogoproto.customtype) = "github.com/gogo/protobuf/test/custom.Uuid", (gogoproto.nullable) = false];
  54. }
  55. given to the face plugin, will generate the following code:
  56. type AFace interface {
  57. Proto() github_com_gogo_protobuf_proto.Message
  58. GetDescription() string
  59. GetNumber() int64
  60. GetId() github_com_gogo_protobuf_test_custom.Uuid
  61. }
  62. func (this *A) Proto() github_com_gogo_protobuf_proto.Message {
  63. return this
  64. }
  65. func (this *A) TestProto() github_com_gogo_protobuf_proto.Message {
  66. return NewAFromFace(this)
  67. }
  68. func (this *A) GetDescription() string {
  69. return this.Description
  70. }
  71. func (this *A) GetNumber() int64 {
  72. return this.Number
  73. }
  74. func (this *A) GetId() github_com_gogo_protobuf_test_custom.Uuid {
  75. return this.Id
  76. }
  77. func NewAFromFace(that AFace) *A {
  78. this := &A{}
  79. this.Description = that.GetDescription()
  80. this.Number = that.GetNumber()
  81. this.Id = that.GetId()
  82. return this
  83. }
  84. and the following test code:
  85. func TestAFace(t *testing7.T) {
  86. popr := math_rand7.New(math_rand7.NewSource(time7.Now().UnixNano()))
  87. p := NewPopulatedA(popr, true)
  88. msg := p.TestProto()
  89. if !p.Equal(msg) {
  90. t.Fatalf("%#v !Face Equal %#v", msg, p)
  91. }
  92. }
  93. The struct A, representing the message, will also be generated just like always.
  94. As you can see A satisfies its own Face, AFace.
  95. Creating another struct which satisfies AFace is very easy.
  96. Simply create all these methods specified in AFace.
  97. Implementing The Proto method is done with the helper function NewAFromFace:
  98. func (this *MyStruct) Proto() proto.Message {
  99. return NewAFromFace(this)
  100. }
  101. just the like TestProto method which is used to test the NewAFromFace function.
  102. */
  103. package face
  104. import (
  105. "github.com/gogo/protobuf/gogoproto"
  106. "github.com/gogo/protobuf/protoc-gen-gogo/generator"
  107. )
  108. type plugin struct {
  109. *generator.Generator
  110. generator.PluginImports
  111. }
  112. func NewPlugin() *plugin {
  113. return &plugin{}
  114. }
  115. func (p *plugin) Name() string {
  116. return "face"
  117. }
  118. func (p *plugin) Init(g *generator.Generator) {
  119. p.Generator = g
  120. }
  121. func (p *plugin) Generate(file *generator.FileDescriptor) {
  122. p.PluginImports = generator.NewPluginImports(p.Generator)
  123. protoPkg := p.NewImport("github.com/gogo/protobuf/proto")
  124. if !gogoproto.ImportsGoGoProto(file.FileDescriptorProto) {
  125. protoPkg = p.NewImport("github.com/golang/protobuf/proto")
  126. }
  127. for _, message := range file.Messages() {
  128. if !gogoproto.IsFace(file.FileDescriptorProto, message.DescriptorProto) {
  129. continue
  130. }
  131. if message.DescriptorProto.GetOptions().GetMapEntry() {
  132. continue
  133. }
  134. if message.DescriptorProto.HasExtension() {
  135. panic("face does not support message with extensions")
  136. }
  137. if gogoproto.HasGoGetters(file.FileDescriptorProto, message.DescriptorProto) {
  138. panic("face requires getters to be disabled please use gogoproto.getters or gogoproto.getters_all and set it to false")
  139. }
  140. ccTypeName := generator.CamelCaseSlice(message.TypeName())
  141. p.P(`type `, ccTypeName, `Face interface{`)
  142. p.In()
  143. p.P(`Proto() `, protoPkg.Use(), `.Message`)
  144. for _, field := range message.Field {
  145. fieldname := p.GetFieldName(message, field)
  146. goTyp, _ := p.GoType(message, field)
  147. if p.IsMap(field) {
  148. m := p.GoMapType(nil, field)
  149. goTyp = m.GoType
  150. }
  151. p.P(`Get`, fieldname, `() `, goTyp)
  152. }
  153. p.Out()
  154. p.P(`}`)
  155. p.P(``)
  156. p.P(`func (this *`, ccTypeName, `) Proto() `, protoPkg.Use(), `.Message {`)
  157. p.In()
  158. p.P(`return this`)
  159. p.Out()
  160. p.P(`}`)
  161. p.P(``)
  162. p.P(`func (this *`, ccTypeName, `) TestProto() `, protoPkg.Use(), `.Message {`)
  163. p.In()
  164. p.P(`return New`, ccTypeName, `FromFace(this)`)
  165. p.Out()
  166. p.P(`}`)
  167. p.P(``)
  168. for _, field := range message.Field {
  169. fieldname := p.GetFieldName(message, field)
  170. goTyp, _ := p.GoType(message, field)
  171. if p.IsMap(field) {
  172. m := p.GoMapType(nil, field)
  173. goTyp = m.GoType
  174. }
  175. p.P(`func (this *`, ccTypeName, `) Get`, fieldname, `() `, goTyp, `{`)
  176. p.In()
  177. p.P(` return this.`, fieldname)
  178. p.Out()
  179. p.P(`}`)
  180. p.P(``)
  181. }
  182. p.P(``)
  183. p.P(`func New`, ccTypeName, `FromFace(that `, ccTypeName, `Face) *`, ccTypeName, ` {`)
  184. p.In()
  185. p.P(`this := &`, ccTypeName, `{}`)
  186. for _, field := range message.Field {
  187. fieldname := p.GetFieldName(message, field)
  188. p.P(`this.`, fieldname, ` = that.Get`, fieldname, `()`)
  189. }
  190. p.P(`return this`)
  191. p.Out()
  192. p.P(`}`)
  193. p.P(``)
  194. }
  195. }
  196. func init() {
  197. generator.RegisterPlugin(NewPlugin())
  198. }