Commit 00b0af08 authored by Jon Skeet's avatar Jon Skeet

Fix default instance behaviour for repeated types

parent 4f435a50
......@@ -3,6 +3,7 @@ using System.Collections.Generic;
using System.Reflection;
using Google.ProtocolBuffers.Collections;
using Google.ProtocolBuffers.DescriptorProtos;
using System.Collections.ObjectModel;
namespace Google.ProtocolBuffers.Descriptors {
public class FieldDescriptor : IndexedDescriptorBase<FieldDescriptorProto, FieldOptions>, IComparable<FieldDescriptor> {
......@@ -124,8 +125,8 @@ namespace Google.ProtocolBuffers.Descriptors {
/// The field's default value. Valid for all types except messages
/// and groups. For all other types, the object returned is of the
/// same class that would be returned by IMessage[this].
/// For repeated fields this will always be an empty list. For message fields it will
/// always be null. For singular values, it will depend on the descriptor.
/// For repeated fields this will always be an empty immutable list compatible with IList[object].
/// For message fields it will always be null. For singular values, it will depend on the descriptor.
/// </value>
public object DefaultValue {
get {
......@@ -383,8 +384,7 @@ namespace Google.ProtocolBuffers.Descriptors {
} else {
// Determine the default default for this field.
if (IsRepeated) {
// FIXME(jonskeet): Find out the correct list type and use that instead.
defaultValue = new List<object>();
defaultValue = Lists<object>.Empty;
} else {
switch (MappedType) {
case MappedType.Enum:
......
......@@ -17,6 +17,8 @@ namespace Google.ProtocolBuffers {
/// FieldSet class, FieldSet just has a MakeImmutable() method. This is safe so long as
/// all callers are careful not to let a mutable FieldSet escape into the open. This would
/// be impossible to guarantee if this were a public class, of course.
///
/// All repeated fields are stored as IList[object] even
/// </summary>
internal class FieldSet {
......@@ -131,7 +133,7 @@ namespace Google.ProtocolBuffers {
int fieldNumber = WireFormat.GetTagFieldNumber(tag);
FieldDescriptor field;
IMessage defaultInstance = null;
IMessage defaultFieldInstance = null;
if (type.IsExtensionNumber(fieldNumber)) {
ExtensionInfo extension = extensionRegistry[type, fieldNumber];
......@@ -139,7 +141,7 @@ namespace Google.ProtocolBuffers {
field = null;
} else {
field = extension.Descriptor;
defaultInstance = extension.DefaultInstance;
defaultFieldInstance = extension.DefaultInstance;
}
} else {
field = type.FindFieldByNumber(fieldNumber);
......@@ -155,8 +157,8 @@ namespace Google.ProtocolBuffers {
case FieldType.Group:
case FieldType.Message: {
IBuilder subBuilder;
if (defaultInstance != null) {
subBuilder = defaultInstance.CreateBuilderForType();
if (defaultFieldInstance != null) {
subBuilder = defaultFieldInstance.CreateBuilderForType();
} else {
subBuilder = builder.CreateBuilderForField(field);
}
......@@ -308,7 +310,8 @@ namespace Google.ProtocolBuffers {
/// <list>
/// <item>For singular message values, null is returned.</item>
/// <item>For singular non-message values, the default value of the field is returned.</item>
/// <item>For repeated values, an empty immutable list is returned.</item>
/// <item>For repeated values, an empty immutable list is returned. This will be compatible
/// with IList[object], regardless of the type of the repeated item.</item>
/// </list>
/// This method returns null if the field is a singular message type
/// and is not set; in this case it is up to the caller to fetch the
......@@ -361,7 +364,7 @@ namespace Google.ProtocolBuffers {
throw new ArgumentException("Indexer specifying field and index can only be called on repeated fields.");
}
return ((List<object>)this[field])[index];
return ((IList<object>) this[field])[index];
}
set {
if (!field.IsRepeated) {
......@@ -372,15 +375,13 @@ namespace Google.ProtocolBuffers {
if (!fields.TryGetValue(field, out list)) {
throw new ArgumentOutOfRangeException();
}
((List<object>) list)[index] = value;
((IList<object>) list)[index] = value;
}
}
/// <summary>
/// See <see cref="IBuilder.AddRepeatedField" />
/// </summary>
/// <param name="field"></param>
/// <param name="value"></param>
internal void AddRepeatedField(FieldDescriptor field, object value) {
if (!field.IsRepeated) {
throw new ArgumentException("AddRepeatedField can only be called on repeated fields.");
......@@ -391,7 +392,7 @@ namespace Google.ProtocolBuffers {
list = new List<object>();
fields[field] = list;
}
((List<object>) list).Add(value);
((IList<object>) list).Add(value);
}
/// <summary>
......@@ -454,7 +455,7 @@ namespace Google.ProtocolBuffers {
throw new ArgumentException("GetRepeatedFieldCount() can only be called on repeated fields.");
}
return ((List<object>) this[field]).Count;
return ((IList<object>) this[field]).Count;
}
/// <summary>
......@@ -480,8 +481,8 @@ namespace Google.ProtocolBuffers {
existingValue = new List<object>();
fields[field] = existingValue;
}
List<object> list = (List<object>)existingValue;
foreach (object otherValue in (IEnumerable)entry.Value) {
IList<object> list = (IList<object>) existingValue;
foreach (object otherValue in (IEnumerable) entry.Value) {
list.Add(otherValue);
}
} else if (field.MappedType == MappedType.Message && existingValue != null) {
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment