Commit c34ed5c9 authored by Jon Skeet's avatar Jon Skeet

Merge pull request #846 from jskeet/tostring

Support ToString in RepeatedField and MapField.
parents 2842568f 9ed6d4da
...@@ -562,6 +562,20 @@ namespace Google.Protobuf.Collections ...@@ -562,6 +562,20 @@ namespace Google.Protobuf.Collections
Assert.IsFalse(values.Contains(null)); Assert.IsFalse(values.Contains(null));
} }
[Test]
public void ToString_StringToString()
{
var map = new MapField<string, string> { { "foo", "bar" }, { "x", "y" } };
Assert.AreEqual("{ \"foo\": \"bar\", \"x\": \"y\" }", map.ToString());
}
[Test]
public void ToString_UnsupportedKeyType()
{
var map = new MapField<byte, string> { { 10, "foo" } };
Assert.Throws<ArgumentException>(() => map.ToString());
}
private static KeyValuePair<TKey, TValue> NewKeyValuePair<TKey, TValue>(TKey key, TValue value) private static KeyValuePair<TKey, TValue> NewKeyValuePair<TKey, TValue>(TKey key, TValue value)
{ {
return new KeyValuePair<TKey, TValue>(key, value); return new KeyValuePair<TKey, TValue>(key, value);
......
...@@ -37,6 +37,7 @@ using System.IO; ...@@ -37,6 +37,7 @@ using System.IO;
using System.Linq; using System.Linq;
using System.Text; using System.Text;
using Google.Protobuf.TestProtos; using Google.Protobuf.TestProtos;
using Google.Protobuf.WellKnownTypes;
using NUnit.Framework; using NUnit.Framework;
namespace Google.Protobuf.Collections namespace Google.Protobuf.Collections
...@@ -599,5 +600,61 @@ namespace Google.Protobuf.Collections ...@@ -599,5 +600,61 @@ namespace Google.Protobuf.Collections
list.Insert(1, "middle"); list.Insert(1, "middle");
CollectionAssert.AreEqual(new[] { "first", "middle", "second" }, list); CollectionAssert.AreEqual(new[] { "first", "middle", "second" }, list);
} }
[Test]
public void ToString_Integers()
{
var list = new RepeatedField<int> { 5, 10, 20 };
var text = list.ToString();
Assert.AreEqual("[ 5, 10, 20 ]", text);
}
[Test]
public void ToString_Strings()
{
var list = new RepeatedField<string> { "x", "y", "z" };
var text = list.ToString();
Assert.AreEqual("[ \"x\", \"y\", \"z\" ]", text);
}
[Test]
public void ToString_Messages()
{
var list = new RepeatedField<TestAllTypes> { new TestAllTypes { SingleDouble = 1.5 }, new TestAllTypes { SingleInt32 = 10 } };
var text = list.ToString();
Assert.AreEqual("[ { \"singleDouble\": 1.5 }, { \"singleInt32\": 10 } ]", text);
}
[Test]
public void ToString_Empty()
{
var list = new RepeatedField<TestAllTypes> { };
var text = list.ToString();
Assert.AreEqual("[ ]", text);
}
[Test]
public void ToString_InvalidElementType()
{
var list = new RepeatedField<decimal> { 15m };
Assert.Throws<ArgumentException>(() => list.ToString());
}
[Test]
public void ToString_Timestamp()
{
var list = new RepeatedField<Timestamp> { Timestamp.FromDateTime(new DateTime(2015, 10, 1, 12, 34, 56, DateTimeKind.Utc)) };
var text = list.ToString();
Assert.AreEqual("[ \"2015-10-01T12:34:56Z\" ]", text);
}
[Test]
public void ToString_Struct()
{
var message = new Struct { Fields = { { "foo", new Value { NumberValue = 20 } } } };
var list = new RepeatedField<Struct> { message };
var text = list.ToString();
Assert.AreEqual(text, "[ { \"foo\": 20 } ]", message.ToString());
}
} }
} }
...@@ -35,6 +35,7 @@ using System; ...@@ -35,6 +35,7 @@ using System;
using System.Collections; using System.Collections;
using System.Collections.Generic; using System.Collections.Generic;
using System.Linq; using System.Linq;
using System.Text;
using Google.Protobuf.Compatibility; using Google.Protobuf.Compatibility;
namespace Google.Protobuf.Collections namespace Google.Protobuf.Collections
...@@ -45,10 +46,17 @@ namespace Google.Protobuf.Collections ...@@ -45,10 +46,17 @@ namespace Google.Protobuf.Collections
/// <typeparam name="TKey">Key type in the map. Must be a type supported by Protocol Buffer map keys.</typeparam> /// <typeparam name="TKey">Key type in the map. Must be a type supported by Protocol Buffer map keys.</typeparam>
/// <typeparam name="TValue">Value type in the map. Must be a type supported by Protocol Buffers.</typeparam> /// <typeparam name="TValue">Value type in the map. Must be a type supported by Protocol Buffers.</typeparam>
/// <remarks> /// <remarks>
/// <para>
/// This implementation preserves insertion order for simplicity of testing /// This implementation preserves insertion order for simplicity of testing
/// code using maps fields. Overwriting an existing entry does not change the /// code using maps fields. Overwriting an existing entry does not change the
/// position of that entry within the map. Equality is not order-sensitive. /// position of that entry within the map. Equality is not order-sensitive.
/// For string keys, the equality comparison is provided by <see cref="StringComparer.Ordinal" />. /// For string keys, the equality comparison is provided by <see cref="StringComparer.Ordinal" />.
/// </para>
/// <para>
/// This implementation does not generally prohibit the use of key/value types which are not
/// supported by Protocol Buffers (e.g. using a key type of <code>byte</code>) but nor does it guarantee
/// that all operations will work in such cases.
/// </para>
/// </remarks> /// </remarks>
public sealed class MapField<TKey, TValue> : IDeepCloneable<MapField<TKey, TValue>>, IDictionary<TKey, TValue>, IEquatable<MapField<TKey, TValue>>, IDictionary public sealed class MapField<TKey, TValue> : IDeepCloneable<MapField<TKey, TValue>>, IDictionary<TKey, TValue>, IEquatable<MapField<TKey, TValue>>, IDictionary
{ {
...@@ -482,6 +490,17 @@ namespace Google.Protobuf.Collections ...@@ -482,6 +490,17 @@ namespace Google.Protobuf.Collections
return size; return size;
} }
/// <summary>
/// Returns a string representation of this repeated field, in the same
/// way as it would be represented by the default JSON formatter.
/// </summary>
public override string ToString()
{
var builder = new StringBuilder();
JsonFormatter.Default.WriteDictionary(builder, this);
return builder.ToString();
}
#region IDictionary explicit interface implementation #region IDictionary explicit interface implementation
void IDictionary.Add(object key, object value) void IDictionary.Add(object key, object value)
{ {
......
...@@ -33,6 +33,7 @@ ...@@ -33,6 +33,7 @@
using System; using System;
using System.Collections; using System.Collections;
using System.Collections.Generic; using System.Collections.Generic;
using System.Text;
using Google.Protobuf.Compatibility; using Google.Protobuf.Compatibility;
namespace Google.Protobuf.Collections namespace Google.Protobuf.Collections
...@@ -41,6 +42,10 @@ namespace Google.Protobuf.Collections ...@@ -41,6 +42,10 @@ namespace Google.Protobuf.Collections
/// The contents of a repeated field: essentially, a collection with some extra /// The contents of a repeated field: essentially, a collection with some extra
/// restrictions (no null values) and capabilities (deep cloning). /// restrictions (no null values) and capabilities (deep cloning).
/// </summary> /// </summary>
/// <remarks>
/// This implementation does not generally prohibit the use of types which are not
/// supported by Protocol Buffers but nor does it guarantee that all operations will work in such cases.
/// </remarks>
/// <typeparam name="T">The element type of the repeated field.</typeparam> /// <typeparam name="T">The element type of the repeated field.</typeparam>
public sealed class RepeatedField<T> : IList<T>, IList, IDeepCloneable<RepeatedField<T>>, IEquatable<RepeatedField<T>> public sealed class RepeatedField<T> : IList<T>, IList, IDeepCloneable<RepeatedField<T>>, IEquatable<RepeatedField<T>>
{ {
...@@ -464,6 +469,17 @@ namespace Google.Protobuf.Collections ...@@ -464,6 +469,17 @@ namespace Google.Protobuf.Collections
array[count] = default(T); array[count] = default(T);
} }
/// <summary>
/// Returns a string representation of this repeated field, in the same
/// way as it would be represented by the default JSON formatter.
/// </summary>
public override string ToString()
{
var builder = new StringBuilder();
JsonFormatter.Default.WriteList(builder, this);
return builder.ToString();
}
/// <summary> /// <summary>
/// Gets or sets the item at the specified index. /// Gets or sets the item at the specified index.
/// </summary> /// </summary>
......
...@@ -170,7 +170,7 @@ namespace Google.Protobuf ...@@ -170,7 +170,7 @@ namespace Google.Protobuf
continue; continue;
} }
// Omit awkward (single) values such as unknown enum values // Omit awkward (single) values such as unknown enum values
if (!field.IsRepeated && !field.IsMap && !CanWriteSingleValue(accessor.Descriptor, value)) if (!field.IsRepeated && !field.IsMap && !CanWriteSingleValue(value))
{ {
continue; continue;
} }
...@@ -182,7 +182,7 @@ namespace Google.Protobuf ...@@ -182,7 +182,7 @@ namespace Google.Protobuf
} }
WriteString(builder, ToCamelCase(accessor.Descriptor.Name)); WriteString(builder, ToCamelCase(accessor.Descriptor.Name));
builder.Append(": "); builder.Append(": ");
WriteValue(builder, accessor, value); WriteValue(builder, value);
first = false; first = false;
} }
builder.Append(first ? "}" : " }"); builder.Append(first ? "}" : " }");
...@@ -291,93 +291,81 @@ namespace Google.Protobuf ...@@ -291,93 +291,81 @@ namespace Google.Protobuf
throw new ArgumentException("Invalid field type"); throw new ArgumentException("Invalid field type");
} }
} }
private void WriteValue(StringBuilder builder, IFieldAccessor accessor, object value) private void WriteValue(StringBuilder builder, object value)
{ {
if (accessor.Descriptor.IsMap) if (value == null)
{ {
WriteDictionary(builder, accessor, (IDictionary) value); WriteNull(builder);
} }
else if (accessor.Descriptor.IsRepeated) else if (value is bool)
{ {
WriteList(builder, accessor, (IList) value); builder.Append((bool) value ? "true" : "false");
} }
else else if (value is ByteString)
{ {
WriteSingleValue(builder, accessor.Descriptor, value); // Nothing in Base64 needs escaping
builder.Append('"');
builder.Append(((ByteString) value).ToBase64());
builder.Append('"');
} }
} else if (value is string)
private void WriteSingleValue(StringBuilder builder, FieldDescriptor descriptor, object value)
{
switch (descriptor.FieldType)
{ {
case FieldType.Bool: WriteString(builder, (string) value);
builder.Append((bool) value ? "true" : "false"); }
break; else if (value is IDictionary)
case FieldType.Bytes: {
// Nothing in Base64 needs escaping WriteDictionary(builder, (IDictionary) value);
}
else if (value is IList)
{
WriteList(builder, (IList) value);
}
else if (value is int || value is uint)
{
IFormattable formattable = (IFormattable) value;
builder.Append(formattable.ToString("d", CultureInfo.InvariantCulture));
}
else if (value is long || value is ulong)
{
builder.Append('"');
IFormattable formattable = (IFormattable) value;
builder.Append(formattable.ToString("d", CultureInfo.InvariantCulture));
builder.Append('"');
}
else if (value is System.Enum)
{
WriteString(builder, value.ToString());
}
else if (value is float || value is double)
{
string text = ((IFormattable) value).ToString("r", CultureInfo.InvariantCulture);
if (text == "NaN" || text == "Infinity" || text == "-Infinity")
{
builder.Append('"'); builder.Append('"');
builder.Append(((ByteString) value).ToBase64()); builder.Append(text);
builder.Append('"'); builder.Append('"');
break; }
case FieldType.String: else
WriteString(builder, (string) value); {
break; builder.Append(text);
case FieldType.Fixed32: }
case FieldType.UInt32: }
case FieldType.SInt32: else if (value is IMessage)
case FieldType.Int32: {
case FieldType.SFixed32: IMessage message = (IMessage) value;
{ if (message.Descriptor.IsWellKnownType)
IFormattable formattable = (IFormattable) value; {
builder.Append(formattable.ToString("d", CultureInfo.InvariantCulture)); WriteWellKnownTypeValue(builder, message.Descriptor, value, true);
break; }
} else
case FieldType.Enum: {
EnumValueDescriptor enumValue = descriptor.EnumType.FindValueByNumber((int) value); WriteMessage(builder, (IMessage) value);
// We will already have validated that this is a known value. }
WriteString(builder, enumValue.Name); }
break; else
case FieldType.Fixed64: {
case FieldType.UInt64: throw new ArgumentException("Unable to format value of type " + value.GetType());
case FieldType.SFixed64:
case FieldType.Int64:
case FieldType.SInt64:
{
builder.Append('"');
IFormattable formattable = (IFormattable) value;
builder.Append(formattable.ToString("d", CultureInfo.InvariantCulture));
builder.Append('"');
break;
}
case FieldType.Double:
case FieldType.Float:
string text = ((IFormattable) value).ToString("r", CultureInfo.InvariantCulture);
if (text == "NaN" || text == "Infinity" || text == "-Infinity")
{
builder.Append('"');
builder.Append(text);
builder.Append('"');
}
else
{
builder.Append(text);
}
break;
case FieldType.Message:
case FieldType.Group: // Never expect to get this, but...
if (descriptor.MessageType.IsWellKnownType)
{
WriteWellKnownTypeValue(builder, descriptor.MessageType, value, true);
}
else
{
WriteMessage(builder, (IMessage) value);
}
break;
default:
throw new ArgumentException("Invalid field type: " + descriptor.FieldType);
} }
} }
...@@ -398,7 +386,7 @@ namespace Google.Protobuf ...@@ -398,7 +386,7 @@ namespace Google.Protobuf
// so we can write it as if we were unconditionally writing the Value field for the wrapper type. // so we can write it as if we were unconditionally writing the Value field for the wrapper type.
if (descriptor.File == Int32Value.Descriptor.File) if (descriptor.File == Int32Value.Descriptor.File)
{ {
WriteSingleValue(builder, descriptor.FindFieldByNumber(1), value); WriteValue(builder, value);
return; return;
} }
if (descriptor.FullName == Timestamp.Descriptor.FullName) if (descriptor.FullName == Timestamp.Descriptor.FullName)
...@@ -424,7 +412,7 @@ namespace Google.Protobuf ...@@ -424,7 +412,7 @@ namespace Google.Protobuf
if (descriptor.FullName == ListValue.Descriptor.FullName) if (descriptor.FullName == ListValue.Descriptor.FullName)
{ {
var fieldAccessor = descriptor.Fields[ListValue.ValuesFieldNumber].Accessor; var fieldAccessor = descriptor.Fields[ListValue.ValuesFieldNumber].Accessor;
WriteList(builder, fieldAccessor, (IList) fieldAccessor.GetValue((IMessage) value)); WriteList(builder, (IList) fieldAccessor.GetValue((IMessage) value));
return; return;
} }
if (descriptor.FullName == Value.Descriptor.FullName) if (descriptor.FullName == Value.Descriptor.FullName)
...@@ -565,7 +553,7 @@ namespace Google.Protobuf ...@@ -565,7 +553,7 @@ namespace Google.Protobuf
case Value.BoolValueFieldNumber: case Value.BoolValueFieldNumber:
case Value.StringValueFieldNumber: case Value.StringValueFieldNumber:
case Value.NumberValueFieldNumber: case Value.NumberValueFieldNumber:
WriteSingleValue(builder, specifiedField, value); WriteValue(builder, value);
return; return;
case Value.StructValueFieldNumber: case Value.StructValueFieldNumber:
case Value.ListValueFieldNumber: case Value.ListValueFieldNumber:
...@@ -581,13 +569,13 @@ namespace Google.Protobuf ...@@ -581,13 +569,13 @@ namespace Google.Protobuf
} }
} }
private void WriteList(StringBuilder builder, IFieldAccessor accessor, IList list) internal void WriteList(StringBuilder builder, IList list)
{ {
builder.Append("[ "); builder.Append("[ ");
bool first = true; bool first = true;
foreach (var value in list) foreach (var value in list)
{ {
if (!CanWriteSingleValue(accessor.Descriptor, value)) if (!CanWriteSingleValue(value))
{ {
continue; continue;
} }
...@@ -595,22 +583,20 @@ namespace Google.Protobuf ...@@ -595,22 +583,20 @@ namespace Google.Protobuf
{ {
builder.Append(", "); builder.Append(", ");
} }
WriteSingleValue(builder, accessor.Descriptor, value); WriteValue(builder, value);
first = false; first = false;
} }
builder.Append(first ? "]" : " ]"); builder.Append(first ? "]" : " ]");
} }
private void WriteDictionary(StringBuilder builder, IFieldAccessor accessor, IDictionary dictionary) internal void WriteDictionary(StringBuilder builder, IDictionary dictionary)
{ {
builder.Append("{ "); builder.Append("{ ");
bool first = true; bool first = true;
FieldDescriptor keyType = accessor.Descriptor.MessageType.FindFieldByNumber(1);
FieldDescriptor valueType = accessor.Descriptor.MessageType.FindFieldByNumber(2);
// This will box each pair. Could use IDictionaryEnumerator, but that's ugly in terms of disposal. // This will box each pair. Could use IDictionaryEnumerator, but that's ugly in terms of disposal.
foreach (DictionaryEntry pair in dictionary) foreach (DictionaryEntry pair in dictionary)
{ {
if (!CanWriteSingleValue(valueType, pair.Value)) if (!CanWriteSingleValue(pair.Value))
{ {
continue; continue;
} }
...@@ -619,32 +605,29 @@ namespace Google.Protobuf ...@@ -619,32 +605,29 @@ namespace Google.Protobuf
builder.Append(", "); builder.Append(", ");
} }
string keyText; string keyText;
switch (keyType.FieldType) if (pair.Key is string)
{ {
case FieldType.String: keyText = (string) pair.Key;
keyText = (string) pair.Key; }
break; else if (pair.Key is bool)
case FieldType.Bool: {
keyText = (bool) pair.Key ? "true" : "false"; keyText = (bool) pair.Key ? "true" : "false";
break; }
case FieldType.Fixed32: else if (pair.Key is int || pair.Key is uint | pair.Key is long || pair.Key is ulong)
case FieldType.Fixed64: {
case FieldType.SFixed32: keyText = ((IFormattable) pair.Key).ToString("d", CultureInfo.InvariantCulture);
case FieldType.SFixed64: }
case FieldType.Int32: else
case FieldType.Int64: {
case FieldType.SInt32: if (pair.Key == null)
case FieldType.SInt64: {
case FieldType.UInt32: throw new ArgumentException("Dictionary has entry with null key");
case FieldType.UInt64: }
keyText = ((IFormattable) pair.Key).ToString("d", CultureInfo.InvariantCulture); throw new ArgumentException("Unhandled dictionary key type: " + pair.Key.GetType());
break;
default:
throw new ArgumentException("Invalid key type: " + keyType.FieldType);
} }
WriteString(builder, keyText); WriteString(builder, keyText);
builder.Append(": "); builder.Append(": ");
WriteSingleValue(builder, valueType, pair.Value); WriteValue(builder, pair.Value);
first = false; first = false;
} }
builder.Append(first ? "}" : " }"); builder.Append(first ? "}" : " }");
...@@ -655,12 +638,11 @@ namespace Google.Protobuf ...@@ -655,12 +638,11 @@ namespace Google.Protobuf
/// Currently only relevant for enums, where unknown values can't be represented. /// Currently only relevant for enums, where unknown values can't be represented.
/// For repeated/map fields, this always returns true. /// For repeated/map fields, this always returns true.
/// </summary> /// </summary>
private bool CanWriteSingleValue(FieldDescriptor descriptor, object value) private bool CanWriteSingleValue(object value)
{ {
if (descriptor.FieldType == FieldType.Enum) if (value is System.Enum)
{ {
EnumValueDescriptor enumValue = descriptor.EnumType.FindValueByNumber((int) value); return System.Enum.IsDefined(value.GetType(), value);
return enumValue != null;
} }
return true; return true;
} }
......
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