using System.Collections;
using System.Collections.Generic;
using System.Runtime.Serialization.Formatters;
using System.ComponentModel.DataAnnotations;
using System.Globalization;
using System.Collections.ObjectModel;
using System.Runtime.Serialization;
using Newtonsoft.Json.Linq;
using Newtonsoft.Json.Converters;
using Newtonsoft.Json.Serialization;
public class ObjectToArrayConverter<T> : JsonConverter
public override bool CanConvert(Type objectType)
return typeof(T) == objectType;
public override void WriteJson(JsonWriter writer, object value, JsonSerializer serializer)
var objectType = value.GetType();
var contract = serializer.ContractResolver.ResolveContract(objectType) as JsonObjectContract;
throw new JsonSerializationException(string.Format("invalid type {0}.", objectType.FullName));
writer.WriteStartArray();
foreach (var property in SerializableProperties(contract))
var propertyValue = property.ValueProvider.GetValue(value);
if (property.Converter != null && property.Converter.CanWrite)
property.Converter.WriteJson(writer, propertyValue, serializer);
serializer.Serialize(writer, propertyValue);
public override object ReadJson(JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer)
var contract = serializer.ContractResolver.ResolveContract(objectType) as JsonObjectContract;
throw new JsonSerializationException(string.Format("invalid type {0}.", objectType.FullName));
if (reader.MoveToContentAndAssert().TokenType == JsonToken.Null)
if (reader.TokenType != JsonToken.StartArray)
throw new JsonSerializationException(string.Format("token {0} was not JsonToken.StartArray", reader.TokenType));
existingValue = existingValue ?? contract.DefaultCreator();
using (var enumerator = SerializableProperties(contract).GetEnumerator())
switch (reader.ReadToContentAndAssert().TokenType)
if (!enumerator.MoveNext())
var property = enumerator.Current;
if (property.Converter != null && property.Converter.CanRead)
propertyValue = property.Converter.ReadJson(reader, property.PropertyType, property.ValueProvider.GetValue(existingValue), serializer);
propertyValue = serializer.Deserialize(reader, property.PropertyType);
property.ValueProvider.SetValue(existingValue, propertyValue);
static IEnumerable<JsonProperty> SerializableProperties(JsonObjectContract contract)
return contract.Properties.Where(p => !p.Ignored && p.Readable && p.Writable);
public static partial class JsonExtensions
public static JsonReader ReadToContentAndAssert(this JsonReader reader)
return reader.ReadAndAssert().MoveToContentAndAssert();
public static JsonReader MoveToContentAndAssert(this JsonReader reader)
throw new ArgumentNullException();
if (reader.TokenType == JsonToken.None)
while (reader.TokenType == JsonToken.Comment)
public static JsonReader ReadAndAssert(this JsonReader reader)
throw new ArgumentNullException();
throw new JsonReaderException("Unexpected end of JSON stream.");
[JsonConverter(typeof(ObjectToArrayConverter<Player>))]
public int UniqueID { get; set; }
public string PlayerDescription { get; set; }
public string ShortPlayerDescription { get { return ""; } }
public string IgnoreMe2 { get; set; }
public class ScoreboardResults
public int timestamp { get; set; }
public int total_players { get; set; }
public int max_score { get; set; }
public Dictionary<string, Player> players { get; set; }
""timestamp"": 1473730993,
internal static void Test()
var inputJson = GetJson();
Console.WriteLine("Input JSON: ");
Console.WriteLine(JToken.Parse(inputJson));
var root = JsonConvert.DeserializeObject<ScoreboardResults>(inputJson);
var json2 = JsonConvert.SerializeObject(root, Formatting.Indented);
Console.WriteLine("\nOutput JSON: ");
Console.WriteLine(json2);
var root2 = JsonConvert.DeserializeObject<ScoreboardResults>(json2);
var json3 = JsonConvert.SerializeObject(root2, Formatting.Indented);
Assert.IsTrue(json2 == json3);
Assert.IsTrue(JToken.Parse(inputJson)
.SelectTokens("players.*")
.SelectTokens("players.*")
(a1, a2) => a1.Count >= a2.Count && a1.Zip(a2, (i1, i2) => JToken.DeepEquals(i1, i2)).All(b => b))
for (int i = 1; i < json3.Length - 1; i++)
Assert.Throws(Is.InstanceOf(typeof(JsonException)), () => JsonConvert.DeserializeObject<ScoreboardResults>(json3.Substring(i, json3.Length - i)));
public static void Main()
Console.WriteLine("Environment version: {0} ({1})", System.Runtime.InteropServices.RuntimeInformation.FrameworkDescription , GetNetCoreVersion());
Console.WriteLine("{0} version: {1}", typeof(JsonSerializer).Assembly.GetName().Name, typeof(JsonSerializer).Assembly.FullName);
Console.WriteLine("Failed with unhandled exception: ");
public static string GetNetCoreVersion()
var assembly = typeof(System.Runtime.GCSettings).GetTypeInfo().Assembly;
var assemblyPath = assembly.CodeBase.Split(new[] { '/', '\\' }, StringSplitOptions.RemoveEmptyEntries);
int netCoreAppIndex = Array.IndexOf(assemblyPath, "Microsoft.NETCore.App");
if (netCoreAppIndex > 0 && netCoreAppIndex < assemblyPath.Length - 2)
return assemblyPath[netCoreAppIndex + 1];