JsonSubTypes 2.0.1
JsonSubTypes
JsonSubTypes is a discriminated Json sub-type Converter implementation for .NET
DeserializeObject with custom type property name
[JsonConverter(typeof(JsonSubtypes), "Kind")]
public interface IAnimal
{
string Kind { get; }
}
public class Dog : IAnimal
{
public string Kind { get; } = "Dog";
public string Breed { get; set; }
}
public class Cat : IAnimal {
public string Kind { get; } = "Cat";
public bool Declawed { get; set;}
}
The second parameter of the JsonConverter
attribute is the JSON property name that will be use to retreive the type information from JSON.
var animal = JsonConvert.DeserializeObject<IAnimal>("{\"Kind\":\"Dog\",\"Breed\":\"Jack Russell Terrier\"}");
Assert.AreEqual("Jack Russell Terrier", (animal as Dog)?.Breed);
N.B.: This only works for types in the same assembly as the base type/interface and either in the same namespace or with a fully qualified type name.
DeserializeObject with custom type mapping
[JsonConverter(typeof(JsonSubtypes), "Sound")]
[JsonSubtypes.KnownSubType(typeof(Dog), "Bark")]
[JsonSubtypes.KnownSubType(typeof(Cat), "Meow")]
public class Animal
{
public virtual string Sound { get; }
public string Color { get; set; }
}
public class Dog : Animal
{
public override string Sound { get; } = "Bark";
public string Breed { get; set; }
}
public class Cat : Animal
{
public override string Sound { get; } = "Meow";
public bool Declawed { get; set; }
}
var animal = JsonConvert.DeserializeObject<IAnimal>("{\"Sound\":\"Bark\",\"Breed\":\"Jack Russell Terrier\"}");
Assert.AreEqual("Jack Russell Terrier", (animal as Dog)?.Breed);
N.B.: Also works with other kind of value than string, i.e.: enums, int, ...
SerializeObject and DeserializeObject with custom type property only present in JSON
This mode of operation only works when JsonSubTypes is explicitely registered in JSON.NET's serializer settings, and not through the [JsonConverter]
attribute.
public abstract class Animal
{
public int Age { get; set; }
}
public class Dog : Animal
{
public bool CanBark { get; set; } = true;
}
public class Cat : Animal
{
public int Lives { get; set; } = 7;
}
public enum AnimalType
{
Dog = 1,
Cat = 2
}
Registration:
var settings = new JsonSerializerSettings();
settings.Converters.Add(JsonSubtypesConverterBuilder
.Of(typeof(Animal), "Type") // type property is only defined here
.RegisterSubtype(typeof(Cat), AnimalType.Cat)
.RegisterSubtype(typeof(Dog), AnimalType.Dog)
.SerializeDiscriminatorProperty() // ask to serialize the type property
.Build());
or using syntax with generics:
var settings = new JsonSerializerSettings();
settings.Converters.Add(JsonSubtypesConverterBuilder
.Of<Animal>("Type") // type property is only defined here
.RegisterSubtype<Cat>(AnimalType.Cat)
.RegisterSubtype<Dog>(AnimalType.Dog)
.SerializeDiscriminatorProperty() // ask to serialize the type property
.Build());
De-/Serialization:
var cat = new Cat { Age = 11, Lives = 6 }
var json = JsonConvert.SerializeObject(cat, settings);
Assert.Equal("{\"Lives\":6,\"Age\":11,\"Type\":2}", json);
var result = JsonConvert.DeserializeObject<Animal>(json, settings);
Assert.Equal(typeof(Cat), result.GetType());
Assert.Equal(11, result.Age);
Assert.Equal(6, (result as Cat)?.Lives);
DeserializeObject mapping by property presence
[JsonConverter(typeof(JsonSubtypes))]
[JsonSubtypes.KnownSubTypeWithProperty(typeof(Employee), "JobTitle")]
[JsonSubtypes.KnownSubTypeWithProperty(typeof(Artist), "Skill")]
public class Person
{
public string FirstName { get; set; }
public string LastName { get; set; }
}
public class Employee : Person
{
public string Department { get; set; }
public string JobTitle { get; set; }
}
public class Artist : Person
{
public string Skill { get; set; }
}
or using syntax with generics:
string json = "[{\"Department\":\"Department1\",\"JobTitle\":\"JobTitle1\",\"FirstName\":\"FirstName1\",\"LastName\":\"LastName1\"}," +
"{\"Department\":\"Department1\",\"JobTitle\":\"JobTitle1\",\"FirstName\":\"FirstName1\",\"LastName\":\"LastName1\"}," +
"{\"Skill\":\"Painter\",\"FirstName\":\"FirstName1\",\"LastName\":\"LastName1\"}]";
var persons = JsonConvert.DeserializeObject<IReadOnlyCollection<Person>>(json);
Assert.AreEqual("Painter", (persons.Last() as Artist)?.Skill);
Registration:
settings.Converters.Add(JsonSubtypesWithPropertyConverterBuilder
.Of(typeof(Person))
.RegisterSubtypeWithProperty(typeof(Employee), "JobTitle")
.RegisterSubtypeWithProperty(typeof(Artist), "Skill")
.Build());
or
settings.Converters.Add(JsonSubtypesWithPropertyConverterBuilder
.Of<Person>()
.RegisterSubtypeWithProperty<Employee>("JobTitle")
.RegisterSubtypeWithProperty<Artist>("Skill")
.Build());
A default class other than the base type can be defined
[JsonConverter(typeof(JsonSubtypes))]
[JsonSubtypes.KnownSubType(typeof(ConstantExpression), "Constant")]
[JsonSubtypes.FallBackSubType(typeof(UnknownExpression))]
public interface IExpression
{
string Type { get; }
}
Or with code configuration:
settings.Converters.Add(JsonSubtypesConverterBuilder
.Of(typeof(IExpression), "Type")
.SetFallbackSubtype(typeof(UnknownExpression))
.RegisterSubtype(typeof(ConstantExpression), "Constant")
.Build());
settings.Converters.Add(JsonSubtypesWithPropertyConverterBuilder
.Of(typeof(IExpression))
.SetFallbackSubtype(typeof(UnknownExpression))
.RegisterSubtype(typeof(ConstantExpression), "Value")
.Build());
💖 Support this project
If this project helped you save money or time or simply makes your life also easier, you can give me a cup of coffee =)
License
Showing the top 20 packages that depend on JsonSubTypes.
Packages | Downloads |
---|---|
Schick.Keycloak.RestApiClient
Keycloak Admin REST API client
|
44 |
Schick.Keycloak.RestApiClient
Keycloak Admin REST API client
|
7 |
Schick.Keycloak.RestApiClient
Keycloak Admin REST API client
|
6 |
Schick.Keycloak.RestApiClient
Keycloak Admin REST API client
|
4 |
Schick.Keycloak.RestApiClient
Time Tracking Report Server
|
3 |
Schick.Keycloak.RestApiClient
Keycloak Admin REST API client
|
3 |
Schick.Keycloak.RestApiClient
Keycloak Admin REST API client
|
2 |
Schick.Keycloak.RestApiClient
Keycloak Admin REST API client
|
1 |
.NET Framework 3.5
- Newtonsoft.Json (>= 13.0.1)
.NET Framework 4.0
- Newtonsoft.Json (>= 13.0.1)
.NET Framework 4.5
- Newtonsoft.Json (>= 13.0.1)
.NET Framework 4.6
- Newtonsoft.Json (>= 13.0.1)
.NET Framework 4.7
- Newtonsoft.Json (>= 13.0.1)
.NET Standard 1.3
- NETStandard.Library (>= 1.6.1)
- Newtonsoft.Json (>= 13.0.1)
.NET Standard 2.0
- Newtonsoft.Json (>= 13.0.1)
Version | Downloads | Last updated |
---|---|---|
2.0.1 | 38 | 08/12/2024 |
2.0.0 | 5 | 09/06/2024 |
1.9.0 | 7 | 08/27/2024 |
1.8.0 | 3 | 09/07/2024 |
1.7.0 | 1 | 09/10/2024 |
1.6.0 | 0 | 06/24/2019 |
1.5.2 | 2 | 09/08/2024 |
1.5.1 | 4 | 08/27/2024 |
1.5.0 | 3 | 08/26/2024 |
1.4.0 | 6 | 09/08/2024 |
1.3.1 | 1 | 11/19/2024 |
1.3.0 | 0 | 01/28/2018 |
1.2.0 | 0 | 11/22/2017 |
1.1.3 | 2 | 09/08/2024 |
1.1.2 | 5 | 09/11/2024 |
1.1.1 | 2 | 09/07/2024 |
1.1.0 | 2 | 08/26/2024 |
1.0.0 | 3 | 08/27/2024 |