description.go 5.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201
  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 description (experimental) plugin generates a Description method for each message.
  30. The Description method returns a populated google_protobuf.FileDescriptorSet struct.
  31. This contains the description of the files used to generate this message.
  32. It is enabled by the following extensions:
  33. - description
  34. - description_all
  35. The description plugin also generates a test given it is enabled using one of the following extensions:
  36. - testgen
  37. - testgen_all
  38. Let us look at:
  39. github.com/gogo/protobuf/test/example/example.proto
  40. Btw all the output can be seen at:
  41. github.com/gogo/protobuf/test/example/*
  42. The following message:
  43. message B {
  44. option (gogoproto.description) = true;
  45. optional A A = 1 [(gogoproto.nullable) = false, (gogoproto.embed) = true];
  46. repeated bytes G = 2 [(gogoproto.customtype) = "github.com/gogo/protobuf/test/custom.Uint128", (gogoproto.nullable) = false];
  47. }
  48. given to the description plugin, will generate the following code:
  49. func (this *B) Description() (desc *google_protobuf.FileDescriptorSet) {
  50. return ExampleDescription()
  51. }
  52. and the following test code:
  53. func TestDescription(t *testing9.T) {
  54. ExampleDescription()
  55. }
  56. The hope is to use this struct in some way instead of reflect.
  57. This package is subject to change, since a use has not been figured out yet.
  58. */
  59. package description
  60. import (
  61. "bytes"
  62. "compress/gzip"
  63. "fmt"
  64. "github.com/gogo/protobuf/gogoproto"
  65. "github.com/gogo/protobuf/proto"
  66. descriptor "github.com/gogo/protobuf/protoc-gen-gogo/descriptor"
  67. "github.com/gogo/protobuf/protoc-gen-gogo/generator"
  68. )
  69. type plugin struct {
  70. *generator.Generator
  71. generator.PluginImports
  72. }
  73. func NewPlugin() *plugin {
  74. return &plugin{}
  75. }
  76. func (p *plugin) Name() string {
  77. return "description"
  78. }
  79. func (p *plugin) Init(g *generator.Generator) {
  80. p.Generator = g
  81. }
  82. func (p *plugin) Generate(file *generator.FileDescriptor) {
  83. used := false
  84. localName := generator.FileName(file)
  85. p.PluginImports = generator.NewPluginImports(p.Generator)
  86. descriptorPkg := p.NewImport("github.com/gogo/protobuf/protoc-gen-gogo/descriptor")
  87. protoPkg := p.NewImport("github.com/gogo/protobuf/proto")
  88. gzipPkg := p.NewImport("compress/gzip")
  89. bytesPkg := p.NewImport("bytes")
  90. ioutilPkg := p.NewImport("io/ioutil")
  91. for _, message := range file.Messages() {
  92. if !gogoproto.HasDescription(file.FileDescriptorProto, message.DescriptorProto) {
  93. continue
  94. }
  95. if message.DescriptorProto.GetOptions().GetMapEntry() {
  96. continue
  97. }
  98. used = true
  99. ccTypeName := generator.CamelCaseSlice(message.TypeName())
  100. p.P(`func (this *`, ccTypeName, `) Description() (desc *`, descriptorPkg.Use(), `.FileDescriptorSet) {`)
  101. p.In()
  102. p.P(`return `, localName, `Description()`)
  103. p.Out()
  104. p.P(`}`)
  105. }
  106. if used {
  107. p.P(`func `, localName, `Description() (desc *`, descriptorPkg.Use(), `.FileDescriptorSet) {`)
  108. p.In()
  109. //Don't generate SourceCodeInfo, since it will create too much code.
  110. ss := make([]*descriptor.SourceCodeInfo, 0)
  111. for _, f := range p.Generator.AllFiles().GetFile() {
  112. ss = append(ss, f.SourceCodeInfo)
  113. f.SourceCodeInfo = nil
  114. }
  115. b, err := proto.Marshal(p.Generator.AllFiles())
  116. if err != nil {
  117. panic(err)
  118. }
  119. for i, f := range p.Generator.AllFiles().GetFile() {
  120. f.SourceCodeInfo = ss[i]
  121. }
  122. p.P(`d := &`, descriptorPkg.Use(), `.FileDescriptorSet{}`)
  123. var buf bytes.Buffer
  124. w, _ := gzip.NewWriterLevel(&buf, gzip.BestCompression)
  125. w.Write(b)
  126. w.Close()
  127. b = buf.Bytes()
  128. p.P("var gzipped = []byte{")
  129. p.In()
  130. p.P("// ", len(b), " bytes of a gzipped FileDescriptorSet")
  131. for len(b) > 0 {
  132. n := 16
  133. if n > len(b) {
  134. n = len(b)
  135. }
  136. s := ""
  137. for _, c := range b[:n] {
  138. s += fmt.Sprintf("0x%02x,", c)
  139. }
  140. p.P(s)
  141. b = b[n:]
  142. }
  143. p.Out()
  144. p.P("}")
  145. p.P(`r := `, bytesPkg.Use(), `.NewReader(gzipped)`)
  146. p.P(`gzipr, err := `, gzipPkg.Use(), `.NewReader(r)`)
  147. p.P(`if err != nil {`)
  148. p.In()
  149. p.P(`panic(err)`)
  150. p.Out()
  151. p.P(`}`)
  152. p.P(`ungzipped, err := `, ioutilPkg.Use(), `.ReadAll(gzipr)`)
  153. p.P(`if err != nil {`)
  154. p.In()
  155. p.P(`panic(err)`)
  156. p.Out()
  157. p.P(`}`)
  158. p.P(`if err := `, protoPkg.Use(), `.Unmarshal(ungzipped, d); err != nil {`)
  159. p.In()
  160. p.P(`panic(err)`)
  161. p.Out()
  162. p.P(`}`)
  163. p.P(`return d`)
  164. p.Out()
  165. p.P(`}`)
  166. }
  167. }
  168. func init() {
  169. generator.RegisterPlugin(NewPlugin())
  170. }