GeneratedBuilder.cs 8.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245
  1. using System;
  2. using System.Collections;
  3. using System.Collections.Generic;
  4. using System.Text;
  5. using Google.ProtocolBuffers.Collections;
  6. using Google.ProtocolBuffers.Descriptors;
  7. using System.IO;
  8. using Google.ProtocolBuffers.FieldAccess;
  9. namespace Google.ProtocolBuffers {
  10. /// <summary>
  11. /// All generated protocol message builder classes extend this class. It implements
  12. /// most of the IBuilder interface using reflection. Users can ignore this class
  13. /// as an implementation detail.
  14. /// </summary>
  15. public abstract class GeneratedBuilder<TMessage, TBuilder> : AbstractBuilder, IBuilder<TMessage>
  16. where TMessage : GeneratedMessage <TMessage, TBuilder>
  17. where TBuilder : GeneratedBuilder<TMessage, TBuilder>, IBuilder<TMessage> {
  18. /// <summary>
  19. /// Returns the message being built at the moment.
  20. /// </summary>
  21. protected abstract TMessage MessageBeingBuilt { get; }
  22. protected internal FieldAccessorTable InternalFieldAccessors {
  23. get { return MessageBeingBuilt.FieldAccesseorsFromBuilder; }
  24. }
  25. public override bool Initialized {
  26. get { return MessageBeingBuilt.IsInitialized; }
  27. }
  28. public override IDictionary<FieldDescriptor, object> AllFields {
  29. get { return MessageBeingBuilt.AllFields; }
  30. }
  31. public override object this[FieldDescriptor field] {
  32. get {
  33. // For repeated fields, the underlying list object is still modifiable at this point.
  34. // Make sure not to expose the modifiable list to the caller.
  35. return field.IsRepeated
  36. ? InternalFieldAccessors[field].GetRepeatedWrapper(this)
  37. : MessageBeingBuilt[field];
  38. }
  39. set {
  40. InternalFieldAccessors[field].SetValue(this, value);
  41. }
  42. }
  43. /// <summary>
  44. /// Adds all of the specified values to the given collection.
  45. /// </summary>
  46. protected void AddRange<T>(IEnumerable<T> source, IList<T> destination) {
  47. List<T> list = destination as List<T>;
  48. if (list != null) {
  49. list.AddRange(source);
  50. } else {
  51. foreach (T element in source) {
  52. destination.Add(element);
  53. }
  54. }
  55. }
  56. /// <summary>
  57. /// Called by derived classes to parse an unknown field.
  58. /// </summary>
  59. /// <returns>true unless the tag is an end-group tag</returns>
  60. protected virtual bool ParseUnknownField(CodedInputStream input, UnknownFieldSet.Builder unknownFields,
  61. ExtensionRegistry extensionRegistry, uint tag) {
  62. return unknownFields.MergeFieldFrom(tag, input);
  63. }
  64. public override MessageDescriptor DescriptorForType {
  65. get { return MessageBeingBuilt.DescriptorForType; }
  66. }
  67. public override int GetRepeatedFieldCount(FieldDescriptor field) {
  68. return MessageBeingBuilt.GetRepeatedFieldCount(field);
  69. }
  70. public override object this[FieldDescriptor field, int index] {
  71. get { return MessageBeingBuilt[field, index]; }
  72. set { InternalFieldAccessors[field].SetRepeated(this, index, value); }
  73. }
  74. public override bool HasField(FieldDescriptor field) {
  75. return MessageBeingBuilt.HasField(field);
  76. }
  77. protected override IMessage BuildImpl() {
  78. return Build();
  79. }
  80. protected override IMessage BuildPartialImpl() {
  81. return BuildPartial();
  82. }
  83. protected override IBuilder CloneImpl() {
  84. return Clone();
  85. }
  86. protected override IMessage DefaultInstanceForTypeImpl {
  87. get { return DefaultInstanceForType; }
  88. }
  89. public override IBuilder CreateBuilderForField(FieldDescriptor field) {
  90. return InternalFieldAccessors[field].CreateBuilder();
  91. }
  92. protected override IBuilder ClearFieldImpl(FieldDescriptor field) {
  93. return ClearField(field);
  94. }
  95. protected override IBuilder AddRepeatedFieldImpl(FieldDescriptor field, object value) {
  96. return AddRepeatedField(field, value);
  97. }
  98. public virtual IBuilder<TMessage> ClearField(FieldDescriptor field) {
  99. InternalFieldAccessors[field].Clear(this);
  100. return this;
  101. }
  102. public virtual IBuilder<TMessage> MergeFrom(TMessage other) {
  103. if (other.DescriptorForType != InternalFieldAccessors.Descriptor) {
  104. throw new ArgumentException("Message type mismatch");
  105. }
  106. foreach (KeyValuePair<FieldDescriptor, object> entry in AllFields) {
  107. FieldDescriptor field = entry.Key;
  108. if (field.IsRepeated) {
  109. // Concatenate repeated fields
  110. foreach (object element in (IEnumerable)entry.Value) {
  111. AddRepeatedField(field, element);
  112. }
  113. } else if (field.MappedType == MappedType.Message && HasField(field)) {
  114. // Merge singular embedded messages
  115. IMessage oldValue = (IMessage)this[field];
  116. this[field] = oldValue.CreateBuilderForType()
  117. .MergeFrom(oldValue)
  118. .MergeFrom((IMessage)entry.Value)
  119. .BuildPartial();
  120. } else {
  121. // Just overwrite
  122. this[field] = entry.Value;
  123. }
  124. }
  125. return this;
  126. }
  127. public virtual IBuilder<TMessage> MergeUnknownFields(UnknownFieldSet unknownFields) {
  128. TMessage result = MessageBeingBuilt;
  129. result.SetUnknownFields(UnknownFieldSet.CreateBuilder(result.UnknownFields)
  130. .MergeFrom(unknownFields)
  131. .Build());
  132. return this;
  133. }
  134. public virtual IBuilder<TMessage> AddRepeatedField(FieldDescriptor field, object value) {
  135. InternalFieldAccessors[field].AddRepeated(this, value);
  136. return this;
  137. }
  138. public IBuilder<TMessage> MergeFrom(ByteString data) {
  139. ((IBuilder) this).MergeFrom(data);
  140. return this;
  141. }
  142. public IBuilder<TMessage> MergeFrom(ByteString data, ExtensionRegistry extensionRegistry) {
  143. ((IBuilder) this).MergeFrom(data, extensionRegistry);
  144. return this;
  145. }
  146. public IBuilder<TMessage> MergeFrom(byte[] data) {
  147. ((IBuilder) this).MergeFrom(data);
  148. return this;
  149. }
  150. public IBuilder<TMessage> MergeFrom(byte[] data, ExtensionRegistry extensionRegistry) {
  151. ((IBuilder) this).MergeFrom(data, extensionRegistry);
  152. return this;
  153. }
  154. public IBuilder<TMessage> MergeFrom(Stream input) {
  155. ((IBuilder) this).MergeFrom(input);
  156. return this;
  157. }
  158. public IBuilder<TMessage> MergeFrom(Stream input, ExtensionRegistry extensionRegistry) {
  159. ((IBuilder) this).MergeFrom(input, extensionRegistry);
  160. return this;
  161. }
  162. /// <summary>
  163. /// Overridden when optimized for speed.
  164. /// </summary>
  165. public virtual IBuilder<TMessage> MergeFrom(CodedInputStream input) {
  166. ((IBuilder)this).MergeFrom(input);
  167. return this;
  168. }
  169. /// <summary>
  170. /// Overridden when optimized for speed.
  171. /// </summary>
  172. public virtual IBuilder<TMessage> MergeFrom(CodedInputStream input, ExtensionRegistry extensionRegistry) {
  173. ((IBuilder)this).MergeFrom(input, extensionRegistry);
  174. return this;
  175. }
  176. /// <summary>
  177. /// Like Build(), but will wrap UninitializedMessageException in
  178. /// InvalidProtocolBufferException.
  179. /// TODO(jonskeet): This used to be generated for each class. Find out why.
  180. /// </summary>
  181. public TMessage BuildParsed() {
  182. if (!Initialized) {
  183. throw new UninitializedMessageException(MessageBeingBuilt).AsInvalidProtocolBufferException();
  184. }
  185. return BuildPartial();
  186. }
  187. /// <summary>
  188. /// Implementation of <see cref="IBuilder{T}.Build" />.
  189. /// TODO(jonskeet): This used to be generated for each class. Find out why.
  190. /// </summary>
  191. public TMessage Build() {
  192. if (!Initialized) {
  193. throw new UninitializedMessageException(MessageBeingBuilt);
  194. }
  195. return BuildPartial();
  196. }
  197. public override UnknownFieldSet UnknownFields {
  198. get { return MessageBeingBuilt.UnknownFields; }
  199. set { MessageBeingBuilt.SetUnknownFields(value); }
  200. }
  201. public abstract TMessage BuildPartial();
  202. public abstract IBuilder<TMessage> Clone();
  203. public abstract new IBuilder<TMessage> Clear();
  204. public abstract TMessage DefaultInstanceForType { get; }
  205. public abstract class ExtendableBuilder : GeneratedBuilder<TMessage, TBuilder> {
  206. }
  207. }
  208. }