ExtensionCollection.cs 4.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105
  1. #region Copyright notice and license
  2. // Protocol Buffers - Google's data interchange format
  3. // Copyright 2008 Google Inc. All rights reserved.
  4. // https://developers.google.com/protocol-buffers/
  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. // * Neither the name of Google Inc. nor the names of its
  17. // contributors may be used to endorse or promote products derived from
  18. // this software without specific prior written permission.
  19. //
  20. // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
  21. // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
  22. // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
  23. // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
  24. // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
  25. // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
  26. // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
  27. // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
  28. // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
  29. // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
  30. // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  31. #endregion
  32. using System.Collections.Generic;
  33. using System.Collections.ObjectModel;
  34. using System.Linq;
  35. namespace Google.Protobuf.Reflection
  36. {
  37. /// <summary>
  38. /// A collection to simplify retrieving the descriptors of extensions in a descriptor for a message
  39. /// </summary>
  40. public sealed class ExtensionCollection
  41. {
  42. private IDictionary<MessageDescriptor, IList<FieldDescriptor>> extensionsByTypeInDeclarationOrder;
  43. private IDictionary<MessageDescriptor, IList<FieldDescriptor>> extensionsByTypeInNumberOrder;
  44. internal ExtensionCollection(FileDescriptor file, Extension[] extensions)
  45. {
  46. UnorderedExtensions = DescriptorUtil.ConvertAndMakeReadOnly(
  47. file.Proto.Extension,
  48. (extension, i) => new FieldDescriptor(extension, file, null, i, null, extensions?[i]));
  49. }
  50. internal ExtensionCollection(MessageDescriptor message, Extension[] extensions)
  51. {
  52. UnorderedExtensions = DescriptorUtil.ConvertAndMakeReadOnly(
  53. message.Proto.Extension,
  54. (extension, i) => new FieldDescriptor(extension, message.File, message, i, null, extensions?[i]));
  55. }
  56. /// <summary>
  57. /// Returns a readonly list of all the extensions defined in this type in
  58. /// the order they were defined in the source .proto file
  59. /// </summary>
  60. public IList<FieldDescriptor> UnorderedExtensions { get; }
  61. /// <summary>
  62. /// Returns a readonly list of all the extensions define in this type that extend
  63. /// the provided descriptor type in the order they were defined in the source .proto file
  64. /// </summary>
  65. public IList<FieldDescriptor> GetExtensionsInDeclarationOrder(MessageDescriptor descriptor)
  66. {
  67. return extensionsByTypeInDeclarationOrder[descriptor];
  68. }
  69. /// <summary>
  70. /// Returns a readonly list of all the extensions define in this type that extend
  71. /// the provided descriptor type in accending field order
  72. /// </summary>
  73. public IList<FieldDescriptor> GetExtensionsInNumberOrder(MessageDescriptor descriptor)
  74. {
  75. return extensionsByTypeInNumberOrder[descriptor];
  76. }
  77. internal void CrossLink()
  78. {
  79. Dictionary<MessageDescriptor, IList<FieldDescriptor>> declarationOrder = new Dictionary<MessageDescriptor, IList<FieldDescriptor>>();
  80. foreach (FieldDescriptor descriptor in UnorderedExtensions)
  81. {
  82. descriptor.CrossLink();
  83. IList<FieldDescriptor> _;
  84. if (!declarationOrder.TryGetValue(descriptor.ExtendeeType, out _))
  85. declarationOrder.Add(descriptor.ExtendeeType, new List<FieldDescriptor>());
  86. declarationOrder[descriptor.ExtendeeType].Add(descriptor);
  87. }
  88. extensionsByTypeInDeclarationOrder = declarationOrder
  89. .ToDictionary(kvp => kvp.Key, kvp => (IList<FieldDescriptor>)new ReadOnlyCollection<FieldDescriptor>(kvp.Value));
  90. extensionsByTypeInNumberOrder = declarationOrder
  91. .ToDictionary(kvp => kvp.Key, kvp => (IList<FieldDescriptor>)new ReadOnlyCollection<FieldDescriptor>(kvp.Value.OrderBy(field => field.FieldNumber).ToArray()));
  92. }
  93. }
  94. }