|
@@ -105,207 +105,6 @@ namespace Google.ProtocolBuffers {
|
|
return fields.ContainsKey(field);
|
|
return fields.ContainsKey(field);
|
|
}
|
|
}
|
|
|
|
|
|
- // TODO(jonskeet): Should this be in UnknownFieldSet.Builder really? Or CodedInputStream?
|
|
|
|
- internal static void MergeFrom(CodedInputStream input, UnknownFieldSet.Builder unknownFields,
|
|
|
|
- ExtensionRegistry extensionRegistry, IBuilder builder) {
|
|
|
|
- while (true) {
|
|
|
|
- uint tag = input.ReadTag();
|
|
|
|
- if (tag == 0) {
|
|
|
|
- break;
|
|
|
|
- }
|
|
|
|
- if (!MergeFieldFrom(input, unknownFields, extensionRegistry,
|
|
|
|
- builder, tag)) {
|
|
|
|
- // end group tag
|
|
|
|
- break;
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- // TODO(jonskeet): Should this be in UnknownFieldSet.Builder really? Or CodedInputStream?
|
|
|
|
- /// <summary>
|
|
|
|
- /// Like <see cref="MergeFrom(CodedInputStream, UnknownFieldSet.Builder, ExtensionRegistry, IBuilder)" />
|
|
|
|
- /// but parses a single field.
|
|
|
|
- /// </summary>
|
|
|
|
- /// <param name="input">The input to read the field from</param>
|
|
|
|
- /// <param name="unknownFields">The set of unknown fields to add the newly-read field to, if it's not a known field</param>
|
|
|
|
- /// <param name="extensionRegistry">Registry to use when an extension field is encountered</param>
|
|
|
|
- /// <param name="builder">Builder to merge field into, if it's a known field</param>
|
|
|
|
- /// <param name="tag">The tag, which should already have been read from the input</param>
|
|
|
|
- /// <returns>true unless the tag is an end-group tag</returns>
|
|
|
|
- internal static bool MergeFieldFrom(CodedInputStream input, UnknownFieldSet.Builder unknownFields,
|
|
|
|
- ExtensionRegistry extensionRegistry, IBuilder builder, uint tag) {
|
|
|
|
- MessageDescriptor type = builder.DescriptorForType;
|
|
|
|
- if (type.Options.MessageSetWireFormat && tag == WireFormat.MessageSetTag.ItemStart) {
|
|
|
|
- MergeMessageSetExtensionFromCodedStream(input, unknownFields, extensionRegistry, builder);
|
|
|
|
- return true;
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- WireFormat.WireType wireType = WireFormat.GetTagWireType(tag);
|
|
|
|
- int fieldNumber = WireFormat.GetTagFieldNumber(tag);
|
|
|
|
-
|
|
|
|
- FieldDescriptor field;
|
|
|
|
- IMessage defaultFieldInstance = null;
|
|
|
|
-
|
|
|
|
- if (type.IsExtensionNumber(fieldNumber)) {
|
|
|
|
- ExtensionInfo extension = extensionRegistry[type, fieldNumber];
|
|
|
|
- if (extension == null) {
|
|
|
|
- field = null;
|
|
|
|
- } else {
|
|
|
|
- field = extension.Descriptor;
|
|
|
|
- defaultFieldInstance = extension.DefaultInstance;
|
|
|
|
- }
|
|
|
|
- } else {
|
|
|
|
- field = type.FindFieldByNumber(fieldNumber);
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- // Unknown field or wrong wire type. Skip.
|
|
|
|
- if (field == null || wireType != WireFormat.FieldTypeToWireFormatMap[field.FieldType]) {
|
|
|
|
- return unknownFields.MergeFieldFrom(tag, input);
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- object value;
|
|
|
|
- switch (field.FieldType) {
|
|
|
|
- case FieldType.Group:
|
|
|
|
- case FieldType.Message: {
|
|
|
|
- IBuilder subBuilder;
|
|
|
|
- if (defaultFieldInstance != null) {
|
|
|
|
- subBuilder = defaultFieldInstance.WeakCreateBuilderForType();
|
|
|
|
- } else {
|
|
|
|
- subBuilder = builder.CreateBuilderForField(field);
|
|
|
|
- }
|
|
|
|
- if (!field.IsRepeated) {
|
|
|
|
- subBuilder.WeakMergeFrom((IMessage) builder[field]);
|
|
|
|
- }
|
|
|
|
- if (field.FieldType == FieldType.Group) {
|
|
|
|
- input.ReadGroup(field.FieldNumber, subBuilder, extensionRegistry);
|
|
|
|
- } else {
|
|
|
|
- input.ReadMessage(subBuilder, extensionRegistry);
|
|
|
|
- }
|
|
|
|
- value = subBuilder.WeakBuild();
|
|
|
|
- break;
|
|
|
|
- }
|
|
|
|
- case FieldType.Enum: {
|
|
|
|
- int rawValue = input.ReadEnum();
|
|
|
|
- value = field.EnumType.FindValueByNumber(rawValue);
|
|
|
|
- // If the number isn't recognized as a valid value for this enum,
|
|
|
|
- // drop it.
|
|
|
|
- if (value == null) {
|
|
|
|
- unknownFields.MergeVarintField(fieldNumber, (ulong) rawValue);
|
|
|
|
- return true;
|
|
|
|
- }
|
|
|
|
- break;
|
|
|
|
- }
|
|
|
|
- default:
|
|
|
|
- value = input.ReadPrimitiveField(field.FieldType);
|
|
|
|
- break;
|
|
|
|
- }
|
|
|
|
- if (field.IsRepeated) {
|
|
|
|
- builder.WeakAddRepeatedField(field, value);
|
|
|
|
- } else {
|
|
|
|
- builder[field] = value;
|
|
|
|
- }
|
|
|
|
- return true;
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- // TODO(jonskeet): Should this be in UnknownFieldSet.Builder really? Or CodedInputStream?
|
|
|
|
- /// <summary>
|
|
|
|
- /// Called by MergeFieldFrom to parse a MessageSet extension.
|
|
|
|
- /// </summary>
|
|
|
|
- private static void MergeMessageSetExtensionFromCodedStream(CodedInputStream input,
|
|
|
|
- UnknownFieldSet.Builder unknownFields,
|
|
|
|
- ExtensionRegistry extensionRegistry,
|
|
|
|
- IBuilder builder) {
|
|
|
|
- MessageDescriptor type = builder.DescriptorForType;
|
|
|
|
-
|
|
|
|
- // The wire format for MessageSet is:
|
|
|
|
- // message MessageSet {
|
|
|
|
- // repeated group Item = 1 {
|
|
|
|
- // required int32 typeId = 2;
|
|
|
|
- // required bytes message = 3;
|
|
|
|
- // }
|
|
|
|
- // }
|
|
|
|
- // "typeId" is the extension's field number. The extension can only be
|
|
|
|
- // a message type, where "message" contains the encoded bytes of that
|
|
|
|
- // message.
|
|
|
|
- //
|
|
|
|
- // In practice, we will probably never see a MessageSet item in which
|
|
|
|
- // the message appears before the type ID, or where either field does not
|
|
|
|
- // appear exactly once. However, in theory such cases are valid, so we
|
|
|
|
- // should be prepared to accept them.
|
|
|
|
-
|
|
|
|
- int typeId = 0;
|
|
|
|
- ByteString rawBytes = null; // If we encounter "message" before "typeId"
|
|
|
|
- IBuilder subBuilder = null;
|
|
|
|
- FieldDescriptor field = null;
|
|
|
|
-
|
|
|
|
- while (true) {
|
|
|
|
- uint tag = input.ReadTag();
|
|
|
|
- if (tag == 0) {
|
|
|
|
- break;
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- if (tag == WireFormat.MessageSetTag.TypeID) {
|
|
|
|
- typeId = input.ReadInt32();
|
|
|
|
- // Zero is not a valid type ID.
|
|
|
|
- if (typeId != 0) {
|
|
|
|
- ExtensionInfo extension = extensionRegistry[type, typeId];
|
|
|
|
- if (extension != null) {
|
|
|
|
- field = extension.Descriptor;
|
|
|
|
- subBuilder = extension.DefaultInstance.WeakCreateBuilderForType();
|
|
|
|
- IMessage originalMessage = (IMessage) builder[field];
|
|
|
|
- if (originalMessage != null) {
|
|
|
|
- subBuilder.WeakMergeFrom(originalMessage);
|
|
|
|
- }
|
|
|
|
- if (rawBytes != null) {
|
|
|
|
- // We already encountered the message. Parse it now.
|
|
|
|
- // TODO(jonskeet): Check this is okay. It's subtly different from the Java, as it doesn't create an input stream from rawBytes.
|
|
|
|
- // In fact, why don't we just call MergeFrom(rawBytes)? And what about the extension registry?
|
|
|
|
- subBuilder.WeakMergeFrom(rawBytes.CreateCodedInput());
|
|
|
|
- rawBytes = null;
|
|
|
|
- }
|
|
|
|
- } else {
|
|
|
|
- // Unknown extension number. If we already saw data, put it
|
|
|
|
- // in rawBytes.
|
|
|
|
- if (rawBytes != null) {
|
|
|
|
- unknownFields.MergeField(typeId,
|
|
|
|
- UnknownField.CreateBuilder()
|
|
|
|
- .AddLengthDelimited(rawBytes)
|
|
|
|
- .Build());
|
|
|
|
- rawBytes = null;
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
- } else if (tag == WireFormat.MessageSetTag.Message) {
|
|
|
|
- if (typeId == 0) {
|
|
|
|
- // We haven't seen a type ID yet, so we have to store the raw bytes for now.
|
|
|
|
- rawBytes = input.ReadBytes();
|
|
|
|
- } else if (subBuilder == null) {
|
|
|
|
- // We don't know how to parse this. Ignore it.
|
|
|
|
- unknownFields.MergeField(typeId,
|
|
|
|
- UnknownField.CreateBuilder()
|
|
|
|
- .AddLengthDelimited(input.ReadBytes())
|
|
|
|
- .Build());
|
|
|
|
- } else {
|
|
|
|
- // We already know the type, so we can parse directly from the input
|
|
|
|
- // with no copying. Hooray!
|
|
|
|
- input.ReadMessage(subBuilder, extensionRegistry);
|
|
|
|
- }
|
|
|
|
- } else {
|
|
|
|
- // Unknown tag. Skip it.
|
|
|
|
- if (!input.SkipField(tag)) {
|
|
|
|
- break; // end of group
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- input.CheckLastTagWas(WireFormat.MessageSetTag.ItemEnd);
|
|
|
|
-
|
|
|
|
- if (subBuilder != null) {
|
|
|
|
- builder[field] = subBuilder.WeakBuild();
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
-
|
|
|
|
/// <summary>
|
|
/// <summary>
|
|
/// Clears all fields.
|
|
/// Clears all fields.
|
|
/// </summary>
|
|
/// </summary>
|