union.go 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209
  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 onlyone plugin generates code for the onlyone extension.
  30. All fields must be nullable and only one of the fields may be set, like a union.
  31. Two methods are generated
  32. GetValue() interface{}
  33. and
  34. SetValue(v interface{}) (set bool)
  35. These provide easier interaction with a onlyone.
  36. The onlyone extension is not called union as this causes compile errors in the C++ generated code.
  37. There can only be one ;)
  38. It is enabled by the following extensions:
  39. - onlyone
  40. - onlyone_all
  41. The onlyone plugin also generates a test given it is enabled using one of the following extensions:
  42. - testgen
  43. - testgen_all
  44. Lets look at:
  45. github.com/gogo/protobuf/test/example/example.proto
  46. Btw all the output can be seen at:
  47. github.com/gogo/protobuf/test/example/*
  48. The following message:
  49. message U {
  50. option (gogoproto.onlyone) = true;
  51. optional A A = 1;
  52. optional B B = 2;
  53. }
  54. given to the onlyone plugin, will generate code which looks a lot like this:
  55. func (this *U) GetValue() interface{} {
  56. if this.A != nil {
  57. return this.A
  58. }
  59. if this.B != nil {
  60. return this.B
  61. }
  62. return nil
  63. }
  64. func (this *U) SetValue(value interface{}) bool {
  65. switch vt := value.(type) {
  66. case *A:
  67. this.A = vt
  68. case *B:
  69. this.B = vt
  70. default:
  71. return false
  72. }
  73. return true
  74. }
  75. and the following test code:
  76. func TestUUnion(t *testing.T) {
  77. popr := math_rand.New(math_rand.NewSource(time.Now().UnixNano()))
  78. p := NewPopulatedU(popr)
  79. v := p.GetValue()
  80. msg := &U{}
  81. if !msg.SetValue(v) {
  82. t.Fatalf("Union: Could not set Value")
  83. }
  84. if !p.Equal(msg) {
  85. t.Fatalf("%#v !Union Equal %#v", msg, p)
  86. }
  87. }
  88. */
  89. package union
  90. import (
  91. "github.com/gogo/protobuf/gogoproto"
  92. "github.com/gogo/protobuf/protoc-gen-gogo/generator"
  93. )
  94. type union struct {
  95. *generator.Generator
  96. generator.PluginImports
  97. }
  98. func NewUnion() *union {
  99. return &union{}
  100. }
  101. func (p *union) Name() string {
  102. return "union"
  103. }
  104. func (p *union) Init(g *generator.Generator) {
  105. p.Generator = g
  106. }
  107. func (p *union) Generate(file *generator.FileDescriptor) {
  108. p.PluginImports = generator.NewPluginImports(p.Generator)
  109. for _, message := range file.Messages() {
  110. if !gogoproto.IsUnion(file.FileDescriptorProto, message.DescriptorProto) {
  111. continue
  112. }
  113. if message.DescriptorProto.HasExtension() {
  114. panic("onlyone does not currently support extensions")
  115. }
  116. if message.DescriptorProto.GetOptions().GetMapEntry() {
  117. continue
  118. }
  119. ccTypeName := generator.CamelCaseSlice(message.TypeName())
  120. p.P(`func (this *`, ccTypeName, `) GetValue() interface{} {`)
  121. p.In()
  122. for _, field := range message.Field {
  123. fieldname := p.GetFieldName(message, field)
  124. if fieldname == "Value" {
  125. panic("cannot have a onlyone message " + ccTypeName + " with a field named Value")
  126. }
  127. p.P(`if this.`, fieldname, ` != nil {`)
  128. p.In()
  129. p.P(`return this.`, fieldname)
  130. p.Out()
  131. p.P(`}`)
  132. }
  133. p.P(`return nil`)
  134. p.Out()
  135. p.P(`}`)
  136. p.P(``)
  137. p.P(`func (this *`, ccTypeName, `) SetValue(value interface{}) bool {`)
  138. p.In()
  139. p.P(`switch vt := value.(type) {`)
  140. p.In()
  141. for _, field := range message.Field {
  142. fieldname := p.GetFieldName(message, field)
  143. goTyp, _ := p.GoType(message, field)
  144. p.P(`case `, goTyp, `:`)
  145. p.In()
  146. p.P(`this.`, fieldname, ` = vt`)
  147. p.Out()
  148. }
  149. p.P(`default:`)
  150. p.In()
  151. for _, field := range message.Field {
  152. fieldname := p.GetFieldName(message, field)
  153. if field.IsMessage() {
  154. goTyp, _ := p.GoType(message, field)
  155. obj := p.ObjectNamed(field.GetTypeName()).(*generator.Descriptor)
  156. if gogoproto.IsUnion(obj.File().FileDescriptorProto, obj.DescriptorProto) {
  157. p.P(`this.`, fieldname, ` = new(`, generator.GoTypeToName(goTyp), `)`)
  158. p.P(`if set := this.`, fieldname, `.SetValue(value); set {`)
  159. p.In()
  160. p.P(`return true`)
  161. p.Out()
  162. p.P(`}`)
  163. p.P(`this.`, fieldname, ` = nil`)
  164. }
  165. }
  166. }
  167. p.P(`return false`)
  168. p.Out()
  169. p.P(`}`)
  170. p.P(`return true`)
  171. p.Out()
  172. p.P(`}`)
  173. }
  174. }
  175. func init() {
  176. generator.RegisterPlugin(NewUnion())
  177. }