compare.go 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526
  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. package compare
  29. import (
  30. "github.com/gogo/protobuf/gogoproto"
  31. "github.com/gogo/protobuf/proto"
  32. descriptor "github.com/gogo/protobuf/protoc-gen-gogo/descriptor"
  33. "github.com/gogo/protobuf/protoc-gen-gogo/generator"
  34. "github.com/gogo/protobuf/vanity"
  35. )
  36. type plugin struct {
  37. *generator.Generator
  38. generator.PluginImports
  39. fmtPkg generator.Single
  40. bytesPkg generator.Single
  41. sortkeysPkg generator.Single
  42. protoPkg generator.Single
  43. }
  44. func NewPlugin() *plugin {
  45. return &plugin{}
  46. }
  47. func (p *plugin) Name() string {
  48. return "compare"
  49. }
  50. func (p *plugin) Init(g *generator.Generator) {
  51. p.Generator = g
  52. }
  53. func (p *plugin) Generate(file *generator.FileDescriptor) {
  54. p.PluginImports = generator.NewPluginImports(p.Generator)
  55. p.fmtPkg = p.NewImport("fmt")
  56. p.bytesPkg = p.NewImport("bytes")
  57. p.sortkeysPkg = p.NewImport("github.com/gogo/protobuf/sortkeys")
  58. p.protoPkg = p.NewImport("github.com/gogo/protobuf/proto")
  59. for _, msg := range file.Messages() {
  60. if msg.DescriptorProto.GetOptions().GetMapEntry() {
  61. continue
  62. }
  63. if gogoproto.HasCompare(file.FileDescriptorProto, msg.DescriptorProto) {
  64. p.generateMessage(file, msg)
  65. }
  66. }
  67. }
  68. func (p *plugin) generateNullableField(fieldname string) {
  69. p.P(`if this.`, fieldname, ` != nil && that1.`, fieldname, ` != nil {`)
  70. p.In()
  71. p.P(`if *this.`, fieldname, ` != *that1.`, fieldname, `{`)
  72. p.In()
  73. p.P(`if *this.`, fieldname, ` < *that1.`, fieldname, `{`)
  74. p.In()
  75. p.P(`return -1`)
  76. p.Out()
  77. p.P(`}`)
  78. p.P(`return 1`)
  79. p.Out()
  80. p.P(`}`)
  81. p.Out()
  82. p.P(`} else if this.`, fieldname, ` != nil {`)
  83. p.In()
  84. p.P(`return 1`)
  85. p.Out()
  86. p.P(`} else if that1.`, fieldname, ` != nil {`)
  87. p.In()
  88. p.P(`return -1`)
  89. p.Out()
  90. p.P(`}`)
  91. }
  92. func (p *plugin) generateMsgNullAndTypeCheck(ccTypeName string) {
  93. p.P(`if that == nil {`)
  94. p.In()
  95. p.P(`if this == nil {`)
  96. p.In()
  97. p.P(`return 0`)
  98. p.Out()
  99. p.P(`}`)
  100. p.P(`return 1`)
  101. p.Out()
  102. p.P(`}`)
  103. p.P(``)
  104. p.P(`that1, ok := that.(*`, ccTypeName, `)`)
  105. p.P(`if !ok {`)
  106. p.In()
  107. p.P(`that2, ok := that.(`, ccTypeName, `)`)
  108. p.P(`if ok {`)
  109. p.In()
  110. p.P(`that1 = &that2`)
  111. p.Out()
  112. p.P(`} else {`)
  113. p.In()
  114. p.P(`return 1`)
  115. p.Out()
  116. p.P(`}`)
  117. p.Out()
  118. p.P(`}`)
  119. p.P(`if that1 == nil {`)
  120. p.In()
  121. p.P(`if this == nil {`)
  122. p.In()
  123. p.P(`return 0`)
  124. p.Out()
  125. p.P(`}`)
  126. p.P(`return 1`)
  127. p.Out()
  128. p.P(`} else if this == nil {`)
  129. p.In()
  130. p.P(`return -1`)
  131. p.Out()
  132. p.P(`}`)
  133. }
  134. func (p *plugin) generateField(file *generator.FileDescriptor, message *generator.Descriptor, field *descriptor.FieldDescriptorProto) {
  135. proto3 := gogoproto.IsProto3(file.FileDescriptorProto)
  136. fieldname := p.GetOneOfFieldName(message, field)
  137. repeated := field.IsRepeated()
  138. ctype := gogoproto.IsCustomType(field)
  139. nullable := gogoproto.IsNullable(field)
  140. // oneof := field.OneofIndex != nil
  141. if !repeated {
  142. if ctype {
  143. if nullable {
  144. p.P(`if that1.`, fieldname, ` == nil {`)
  145. p.In()
  146. p.P(`if this.`, fieldname, ` != nil {`)
  147. p.In()
  148. p.P(`return 1`)
  149. p.Out()
  150. p.P(`}`)
  151. p.Out()
  152. p.P(`} else if this.`, fieldname, ` == nil {`)
  153. p.In()
  154. p.P(`return -1`)
  155. p.Out()
  156. p.P(`} else if c := this.`, fieldname, `.Compare(*that1.`, fieldname, `); c != 0 {`)
  157. } else {
  158. p.P(`if c := this.`, fieldname, `.Compare(that1.`, fieldname, `); c != 0 {`)
  159. }
  160. p.In()
  161. p.P(`return c`)
  162. p.Out()
  163. p.P(`}`)
  164. } else {
  165. if field.IsMessage() || p.IsGroup(field) {
  166. if nullable {
  167. p.P(`if c := this.`, fieldname, `.Compare(that1.`, fieldname, `); c != 0 {`)
  168. } else {
  169. p.P(`if c := this.`, fieldname, `.Compare(&that1.`, fieldname, `); c != 0 {`)
  170. }
  171. p.In()
  172. p.P(`return c`)
  173. p.Out()
  174. p.P(`}`)
  175. } else if field.IsBytes() {
  176. p.P(`if c := `, p.bytesPkg.Use(), `.Compare(this.`, fieldname, `, that1.`, fieldname, `); c != 0 {`)
  177. p.In()
  178. p.P(`return c`)
  179. p.Out()
  180. p.P(`}`)
  181. } else if field.IsString() {
  182. if nullable && !proto3 {
  183. p.generateNullableField(fieldname)
  184. } else {
  185. p.P(`if this.`, fieldname, ` != that1.`, fieldname, `{`)
  186. p.In()
  187. p.P(`if this.`, fieldname, ` < that1.`, fieldname, `{`)
  188. p.In()
  189. p.P(`return -1`)
  190. p.Out()
  191. p.P(`}`)
  192. p.P(`return 1`)
  193. p.Out()
  194. p.P(`}`)
  195. }
  196. } else if field.IsBool() {
  197. if nullable && !proto3 {
  198. p.P(`if this.`, fieldname, ` != nil && that1.`, fieldname, ` != nil {`)
  199. p.In()
  200. p.P(`if *this.`, fieldname, ` != *that1.`, fieldname, `{`)
  201. p.In()
  202. p.P(`if !*this.`, fieldname, ` {`)
  203. p.In()
  204. p.P(`return -1`)
  205. p.Out()
  206. p.P(`}`)
  207. p.P(`return 1`)
  208. p.Out()
  209. p.P(`}`)
  210. p.Out()
  211. p.P(`} else if this.`, fieldname, ` != nil {`)
  212. p.In()
  213. p.P(`return 1`)
  214. p.Out()
  215. p.P(`} else if that1.`, fieldname, ` != nil {`)
  216. p.In()
  217. p.P(`return -1`)
  218. p.Out()
  219. p.P(`}`)
  220. } else {
  221. p.P(`if this.`, fieldname, ` != that1.`, fieldname, `{`)
  222. p.In()
  223. p.P(`if !this.`, fieldname, ` {`)
  224. p.In()
  225. p.P(`return -1`)
  226. p.Out()
  227. p.P(`}`)
  228. p.P(`return 1`)
  229. p.Out()
  230. p.P(`}`)
  231. }
  232. } else {
  233. if nullable && !proto3 {
  234. p.generateNullableField(fieldname)
  235. } else {
  236. p.P(`if this.`, fieldname, ` != that1.`, fieldname, `{`)
  237. p.In()
  238. p.P(`if this.`, fieldname, ` < that1.`, fieldname, `{`)
  239. p.In()
  240. p.P(`return -1`)
  241. p.Out()
  242. p.P(`}`)
  243. p.P(`return 1`)
  244. p.Out()
  245. p.P(`}`)
  246. }
  247. }
  248. }
  249. } else {
  250. p.P(`if len(this.`, fieldname, `) != len(that1.`, fieldname, `) {`)
  251. p.In()
  252. p.P(`if len(this.`, fieldname, `) < len(that1.`, fieldname, `) {`)
  253. p.In()
  254. p.P(`return -1`)
  255. p.Out()
  256. p.P(`}`)
  257. p.P(`return 1`)
  258. p.Out()
  259. p.P(`}`)
  260. p.P(`for i := range this.`, fieldname, ` {`)
  261. p.In()
  262. if ctype {
  263. p.P(`if c := this.`, fieldname, `[i].Compare(that1.`, fieldname, `[i]); c != 0 {`)
  264. p.In()
  265. p.P(`return c`)
  266. p.Out()
  267. p.P(`}`)
  268. } else {
  269. if p.IsMap(field) {
  270. m := p.GoMapType(nil, field)
  271. valuegoTyp, _ := p.GoType(nil, m.ValueField)
  272. valuegoAliasTyp, _ := p.GoType(nil, m.ValueAliasField)
  273. nullable, valuegoTyp, valuegoAliasTyp = generator.GoMapValueTypes(field, m.ValueField, valuegoTyp, valuegoAliasTyp)
  274. mapValue := m.ValueAliasField
  275. if mapValue.IsMessage() || p.IsGroup(mapValue) {
  276. if nullable && valuegoTyp == valuegoAliasTyp {
  277. p.P(`if c := this.`, fieldname, `[i].Compare(that1.`, fieldname, `[i]); c != 0 {`)
  278. } else {
  279. // Compare() has a pointer receiver, but map value is a value type
  280. a := `this.` + fieldname + `[i]`
  281. b := `that1.` + fieldname + `[i]`
  282. if valuegoTyp != valuegoAliasTyp {
  283. // cast back to the type that has the generated methods on it
  284. a = `(` + valuegoTyp + `)(` + a + `)`
  285. b = `(` + valuegoTyp + `)(` + b + `)`
  286. }
  287. p.P(`a := `, a)
  288. p.P(`b := `, b)
  289. if nullable {
  290. p.P(`if c := a.Compare(b); c != 0 {`)
  291. } else {
  292. p.P(`if c := (&a).Compare(&b); c != 0 {`)
  293. }
  294. }
  295. p.In()
  296. p.P(`return c`)
  297. p.Out()
  298. p.P(`}`)
  299. } else if mapValue.IsBytes() {
  300. p.P(`if c := `, p.bytesPkg.Use(), `.Compare(this.`, fieldname, `[i], that1.`, fieldname, `[i]); c != 0 {`)
  301. p.In()
  302. p.P(`return c`)
  303. p.Out()
  304. p.P(`}`)
  305. } else if mapValue.IsString() {
  306. p.P(`if this.`, fieldname, `[i] != that1.`, fieldname, `[i] {`)
  307. p.In()
  308. p.P(`if this.`, fieldname, `[i] < that1.`, fieldname, `[i] {`)
  309. p.In()
  310. p.P(`return -1`)
  311. p.Out()
  312. p.P(`}`)
  313. p.P(`return 1`)
  314. p.Out()
  315. p.P(`}`)
  316. } else {
  317. p.P(`if this.`, fieldname, `[i] != that1.`, fieldname, `[i] {`)
  318. p.In()
  319. p.P(`if this.`, fieldname, `[i] < that1.`, fieldname, `[i] {`)
  320. p.In()
  321. p.P(`return -1`)
  322. p.Out()
  323. p.P(`}`)
  324. p.P(`return 1`)
  325. p.Out()
  326. p.P(`}`)
  327. }
  328. } else if field.IsMessage() || p.IsGroup(field) {
  329. if nullable {
  330. p.P(`if c := this.`, fieldname, `[i].Compare(that1.`, fieldname, `[i]); c != 0 {`)
  331. p.In()
  332. p.P(`return c`)
  333. p.Out()
  334. p.P(`}`)
  335. } else {
  336. p.P(`if c := this.`, fieldname, `[i].Compare(&that1.`, fieldname, `[i]); c != 0 {`)
  337. p.In()
  338. p.P(`return c`)
  339. p.Out()
  340. p.P(`}`)
  341. }
  342. } else if field.IsBytes() {
  343. p.P(`if c := `, p.bytesPkg.Use(), `.Compare(this.`, fieldname, `[i], that1.`, fieldname, `[i]); c != 0 {`)
  344. p.In()
  345. p.P(`return c`)
  346. p.Out()
  347. p.P(`}`)
  348. } else if field.IsString() {
  349. p.P(`if this.`, fieldname, `[i] != that1.`, fieldname, `[i] {`)
  350. p.In()
  351. p.P(`if this.`, fieldname, `[i] < that1.`, fieldname, `[i] {`)
  352. p.In()
  353. p.P(`return -1`)
  354. p.Out()
  355. p.P(`}`)
  356. p.P(`return 1`)
  357. p.Out()
  358. p.P(`}`)
  359. } else if field.IsBool() {
  360. p.P(`if this.`, fieldname, `[i] != that1.`, fieldname, `[i] {`)
  361. p.In()
  362. p.P(`if !this.`, fieldname, `[i] {`)
  363. p.In()
  364. p.P(`return -1`)
  365. p.Out()
  366. p.P(`}`)
  367. p.P(`return 1`)
  368. p.Out()
  369. p.P(`}`)
  370. } else {
  371. p.P(`if this.`, fieldname, `[i] != that1.`, fieldname, `[i] {`)
  372. p.In()
  373. p.P(`if this.`, fieldname, `[i] < that1.`, fieldname, `[i] {`)
  374. p.In()
  375. p.P(`return -1`)
  376. p.Out()
  377. p.P(`}`)
  378. p.P(`return 1`)
  379. p.Out()
  380. p.P(`}`)
  381. }
  382. }
  383. p.Out()
  384. p.P(`}`)
  385. }
  386. }
  387. func (p *plugin) generateMessage(file *generator.FileDescriptor, message *generator.Descriptor) {
  388. ccTypeName := generator.CamelCaseSlice(message.TypeName())
  389. p.P(`func (this *`, ccTypeName, `) Compare(that interface{}) int {`)
  390. p.In()
  391. p.generateMsgNullAndTypeCheck(ccTypeName)
  392. oneofs := make(map[string]struct{})
  393. for _, field := range message.Field {
  394. oneof := field.OneofIndex != nil
  395. if oneof {
  396. fieldname := p.GetFieldName(message, field)
  397. if _, ok := oneofs[fieldname]; ok {
  398. continue
  399. } else {
  400. oneofs[fieldname] = struct{}{}
  401. }
  402. p.P(`if that1.`, fieldname, ` == nil {`)
  403. p.In()
  404. p.P(`if this.`, fieldname, ` != nil {`)
  405. p.In()
  406. p.P(`return 1`)
  407. p.Out()
  408. p.P(`}`)
  409. p.Out()
  410. p.P(`} else if this.`, fieldname, ` == nil {`)
  411. p.In()
  412. p.P(`return -1`)
  413. p.Out()
  414. p.P(`} else if c := this.`, fieldname, `.Compare(that1.`, fieldname, `); c != 0 {`)
  415. p.In()
  416. p.P(`return c`)
  417. p.Out()
  418. p.P(`}`)
  419. } else {
  420. p.generateField(file, message, field)
  421. }
  422. }
  423. if message.DescriptorProto.HasExtension() {
  424. if gogoproto.HasExtensionsMap(file.FileDescriptorProto, message.DescriptorProto) {
  425. p.P(`thismap := `, p.protoPkg.Use(), `.GetUnsafeExtensionsMap(this)`)
  426. p.P(`thatmap := `, p.protoPkg.Use(), `.GetUnsafeExtensionsMap(that1)`)
  427. p.P(`extkeys := make([]int32, 0, len(thismap)+len(thatmap))`)
  428. p.P(`for k, _ := range thismap {`)
  429. p.In()
  430. p.P(`extkeys = append(extkeys, k)`)
  431. p.Out()
  432. p.P(`}`)
  433. p.P(`for k, _ := range thatmap {`)
  434. p.In()
  435. p.P(`if _, ok := thismap[k]; !ok {`)
  436. p.In()
  437. p.P(`extkeys = append(extkeys, k)`)
  438. p.Out()
  439. p.P(`}`)
  440. p.Out()
  441. p.P(`}`)
  442. p.P(p.sortkeysPkg.Use(), `.Int32s(extkeys)`)
  443. p.P(`for _, k := range extkeys {`)
  444. p.In()
  445. p.P(`if v, ok := thismap[k]; ok {`)
  446. p.In()
  447. p.P(`if v2, ok := thatmap[k]; ok {`)
  448. p.In()
  449. p.P(`if c := v.Compare(&v2); c != 0 {`)
  450. p.In()
  451. p.P(`return c`)
  452. p.Out()
  453. p.P(`}`)
  454. p.Out()
  455. p.P(`} else {`)
  456. p.In()
  457. p.P(`return 1`)
  458. p.Out()
  459. p.P(`}`)
  460. p.Out()
  461. p.P(`} else {`)
  462. p.In()
  463. p.P(`return -1`)
  464. p.Out()
  465. p.P(`}`)
  466. p.Out()
  467. p.P(`}`)
  468. } else {
  469. fieldname := "XXX_extensions"
  470. p.P(`if c := `, p.bytesPkg.Use(), `.Compare(this.`, fieldname, `, that1.`, fieldname, `); c != 0 {`)
  471. p.In()
  472. p.P(`return c`)
  473. p.Out()
  474. p.P(`}`)
  475. }
  476. }
  477. if gogoproto.HasUnrecognized(file.FileDescriptorProto, message.DescriptorProto) {
  478. fieldname := "XXX_unrecognized"
  479. p.P(`if c := `, p.bytesPkg.Use(), `.Compare(this.`, fieldname, `, that1.`, fieldname, `); c != 0 {`)
  480. p.In()
  481. p.P(`return c`)
  482. p.Out()
  483. p.P(`}`)
  484. }
  485. p.P(`return 0`)
  486. p.Out()
  487. p.P(`}`)
  488. //Generate Compare methods for oneof fields
  489. m := proto.Clone(message.DescriptorProto).(*descriptor.DescriptorProto)
  490. for _, field := range m.Field {
  491. oneof := field.OneofIndex != nil
  492. if !oneof {
  493. continue
  494. }
  495. ccTypeName := p.OneOfTypeName(message, field)
  496. p.P(`func (this *`, ccTypeName, `) Compare(that interface{}) int {`)
  497. p.In()
  498. p.generateMsgNullAndTypeCheck(ccTypeName)
  499. vanity.TurnOffNullableForNativeTypes(field)
  500. p.generateField(file, message, field)
  501. p.P(`return 0`)
  502. p.Out()
  503. p.P(`}`)
  504. }
  505. }
  506. func init() {
  507. generator.RegisterPlugin(NewPlugin())
  508. }