GeneratedBuilder.cs 6.3 KB

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