embedcheck.go 5.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200
  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 embedcheck plugin is used to check whether embed is not used incorrectly.
  30. For instance:
  31. An embedded message has a generated string method, but the is a member of a message which does not.
  32. This causes a warning.
  33. An error is caused by a namespace conflict.
  34. It is enabled by the following extensions:
  35. - embed
  36. - embed_all
  37. For incorrect usage of embed with tests see:
  38. github.com/gogo/protobuf/test/embedconflict
  39. */
  40. package embedcheck
  41. import (
  42. "fmt"
  43. "os"
  44. "github.com/gogo/protobuf/gogoproto"
  45. "github.com/gogo/protobuf/protoc-gen-gogo/generator"
  46. )
  47. type plugin struct {
  48. *generator.Generator
  49. }
  50. func NewPlugin() *plugin {
  51. return &plugin{}
  52. }
  53. func (p *plugin) Name() string {
  54. return "embedcheck"
  55. }
  56. func (p *plugin) Init(g *generator.Generator) {
  57. p.Generator = g
  58. }
  59. var overwriters []map[string]gogoproto.EnableFunc = []map[string]gogoproto.EnableFunc{
  60. {
  61. "stringer": gogoproto.IsStringer,
  62. },
  63. {
  64. "gostring": gogoproto.HasGoString,
  65. },
  66. {
  67. "equal": gogoproto.HasEqual,
  68. },
  69. {
  70. "verboseequal": gogoproto.HasVerboseEqual,
  71. },
  72. {
  73. "size": gogoproto.IsSizer,
  74. "protosizer": gogoproto.IsProtoSizer,
  75. },
  76. {
  77. "unmarshaler": gogoproto.IsUnmarshaler,
  78. "unsafe_unmarshaler": gogoproto.IsUnsafeUnmarshaler,
  79. },
  80. {
  81. "marshaler": gogoproto.IsMarshaler,
  82. "unsafe_marshaler": gogoproto.IsUnsafeMarshaler,
  83. },
  84. }
  85. func (p *plugin) Generate(file *generator.FileDescriptor) {
  86. for _, msg := range file.Messages() {
  87. for _, os := range overwriters {
  88. possible := true
  89. for _, overwriter := range os {
  90. if overwriter(file.FileDescriptorProto, msg.DescriptorProto) {
  91. possible = false
  92. }
  93. }
  94. if possible {
  95. p.checkOverwrite(msg, os)
  96. }
  97. }
  98. p.checkNameSpace(msg)
  99. for _, field := range msg.GetField() {
  100. if gogoproto.IsEmbed(field) && gogoproto.IsCustomName(field) {
  101. fmt.Fprintf(os.Stderr, "ERROR: field %v with custom name %v cannot be embedded", *field.Name, gogoproto.GetCustomName(field))
  102. os.Exit(1)
  103. }
  104. }
  105. p.checkRepeated(msg)
  106. }
  107. for _, e := range file.GetExtension() {
  108. if gogoproto.IsEmbed(e) {
  109. fmt.Fprintf(os.Stderr, "ERROR: extended field %v cannot be embedded", generator.CamelCase(*e.Name))
  110. os.Exit(1)
  111. }
  112. }
  113. }
  114. func (p *plugin) checkNameSpace(message *generator.Descriptor) map[string]bool {
  115. ccTypeName := generator.CamelCaseSlice(message.TypeName())
  116. names := make(map[string]bool)
  117. for _, field := range message.Field {
  118. fieldname := generator.CamelCase(*field.Name)
  119. if field.IsMessage() && gogoproto.IsEmbed(field) {
  120. desc := p.ObjectNamed(field.GetTypeName())
  121. moreNames := p.checkNameSpace(desc.(*generator.Descriptor))
  122. for another := range moreNames {
  123. if names[another] {
  124. fmt.Fprintf(os.Stderr, "ERROR: duplicate embedded fieldname %v in type %v\n", fieldname, ccTypeName)
  125. os.Exit(1)
  126. }
  127. names[another] = true
  128. }
  129. } else {
  130. if names[fieldname] {
  131. fmt.Fprintf(os.Stderr, "ERROR: duplicate embedded fieldname %v in type %v\n", fieldname, ccTypeName)
  132. os.Exit(1)
  133. }
  134. names[fieldname] = true
  135. }
  136. }
  137. return names
  138. }
  139. func (p *plugin) checkOverwrite(message *generator.Descriptor, enablers map[string]gogoproto.EnableFunc) {
  140. ccTypeName := generator.CamelCaseSlice(message.TypeName())
  141. names := []string{}
  142. for name := range enablers {
  143. names = append(names, name)
  144. }
  145. for _, field := range message.Field {
  146. if field.IsMessage() && gogoproto.IsEmbed(field) {
  147. fieldname := generator.CamelCase(*field.Name)
  148. desc := p.ObjectNamed(field.GetTypeName())
  149. msg := desc.(*generator.Descriptor)
  150. for errStr, enabled := range enablers {
  151. if enabled(msg.File().FileDescriptorProto, msg.DescriptorProto) {
  152. fmt.Fprintf(os.Stderr, "WARNING: found non-%v %v with embedded %v %v\n", names, ccTypeName, errStr, fieldname)
  153. }
  154. }
  155. p.checkOverwrite(msg, enablers)
  156. }
  157. }
  158. }
  159. func (p *plugin) checkRepeated(message *generator.Descriptor) {
  160. ccTypeName := generator.CamelCaseSlice(message.TypeName())
  161. for _, field := range message.Field {
  162. if !gogoproto.IsEmbed(field) {
  163. continue
  164. }
  165. if field.IsBytes() {
  166. fieldname := generator.CamelCase(*field.Name)
  167. fmt.Fprintf(os.Stderr, "ERROR: found embedded bytes field %s in message %s\n", fieldname, ccTypeName)
  168. os.Exit(1)
  169. }
  170. if !field.IsRepeated() {
  171. continue
  172. }
  173. fieldname := generator.CamelCase(*field.Name)
  174. fmt.Fprintf(os.Stderr, "ERROR: found repeated embedded field %s in message %s\n", fieldname, ccTypeName)
  175. os.Exit(1)
  176. }
  177. }
  178. func (p *plugin) GenerateImports(*generator.FileDescriptor) {}
  179. func init() {
  180. generator.RegisterPlugin(NewPlugin())
  181. }