using System.Collections;
using System.Collections.Generic;
using System.Globalization;
using Newtonsoft.Json.Linq;
using Newtonsoft.Json.Converters;
using Newtonsoft.Json.Serialization;
public Property Properties { get; set; }
public string MyFirstProp { get; set; }
public string MySecondProp {get; set; }
public class ObjectAsObjectArrayConverter<TObject> : JsonConverter<TObject>
public override void WriteJson(JsonWriter writer, TObject value, JsonSerializer serializer)
var contract = (serializer.ContractResolver.ResolveContract(value.GetType()) as JsonObjectContract) ?? throw new ArgumentException("Wrong contract type");
writer.WriteStartArray();
foreach (var property in contract.Properties.Where(p => ShouldSerialize(p, value)))
var propertyValue = property.ValueProvider.GetValue(value);
if (propertyValue == null && (serializer.NullValueHandling == NullValueHandling.Ignore || property.NullValueHandling == NullValueHandling.Ignore))
writer.WriteStartObject();
writer.WritePropertyName(property.PropertyName);
if (propertyValue == null)
else if (property.Converter != null && property.Converter.CanWrite)
property.Converter.WriteJson(writer, propertyValue, serializer);
serializer.Serialize(writer, propertyValue);
protected virtual bool ShouldSerialize(JsonProperty property, object value) =>
property.Readable && !property.Ignored && (property.ShouldSerialize == null || property.ShouldSerialize(value));
public override TObject ReadJson(JsonReader reader, Type objectType, TObject existingValue, bool hasExistingValue, JsonSerializer serializer)
if (existingValue == null)
existingValue = (TObject)serializer.ContractResolver.ResolveContract(objectType).DefaultCreator();
switch (reader.MoveToContentAndAssert().TokenType)
return (TObject)(object)null;
case JsonToken.StartArray:
while (reader.ReadToContentAndAssert().TokenType != JsonToken.EndArray)
switch (reader.TokenType)
case JsonToken.StartObject:
serializer.Populate(reader, existingValue);
throw new JsonSerializationException("Unexpected token type " + reader.TokenType.ToString());
case JsonToken.StartObject:
serializer.Populate(reader, existingValue);
throw new JsonSerializationException("Unexpected token type " + reader.TokenType.ToString());
public static partial class JsonExtensions
public static JsonReader ReadToContentAndAssert(this JsonReader reader) =>
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.");
public static void Test()
var myClass = new RootObject
var settings = new JsonSerializerSettings
Converters = { new ObjectAsObjectArrayConverter<Property>() },
var json = JsonConvert.SerializeObject(myClass, Formatting.Indented, settings);
var myClass2 = JsonConvert.DeserializeObject<RootObject>(json, settings);
Assert.AreEqual(myClass?.Properties?.MyFirstProp, myClass2?.Properties?.MyFirstProp);
Assert.AreEqual(myClass?.Properties?.MySecondProp, myClass2?.Properties?.MySecondProp);
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.Location.Split(new[] { '/', '\\' }, StringSplitOptions.RemoveEmptyEntries);
int netCoreAppIndex = Array.IndexOf(assemblyPath, "Microsoft.NETCore.App");
if (netCoreAppIndex > 0 && netCoreAppIndex < assemblyPath.Length - 2)
return assemblyPath[netCoreAppIndex + 1];