Properly handle null structs in json
This commit is contained in:
parent
800c039612
commit
ac790cd77b
|
@ -8,37 +8,38 @@ namespace MediaBrowser.Common.Json.Converters
|
||||||
/// Converts a nullable struct or value to/from JSON.
|
/// Converts a nullable struct or value to/from JSON.
|
||||||
/// Required - some clients send an empty string.
|
/// Required - some clients send an empty string.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
/// <typeparam name="T">The struct type.</typeparam>
|
/// <typeparam name="TStruct">The struct type.</typeparam>
|
||||||
public class JsonNullableStructConverter<T> : JsonConverter<T?>
|
public class JsonNullableStructConverter<TStruct> : JsonConverter<TStruct?>
|
||||||
where T : struct
|
where TStruct : struct
|
||||||
{
|
{
|
||||||
private readonly JsonConverter<T?> _baseJsonConverter;
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Initializes a new instance of the <see cref="JsonNullableStructConverter{T}"/> class.
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="baseJsonConverter">The base json converter.</param>
|
|
||||||
public JsonNullableStructConverter(JsonConverter<T?> baseJsonConverter)
|
|
||||||
{
|
|
||||||
_baseJsonConverter = baseJsonConverter;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <inheritdoc />
|
/// <inheritdoc />
|
||||||
public override T? Read(ref Utf8JsonReader reader, Type typeToConvert, JsonSerializerOptions options)
|
public override TStruct? Read(ref Utf8JsonReader reader, Type typeToConvert, JsonSerializerOptions options)
|
||||||
{
|
{
|
||||||
// Handle empty string.
|
if (reader.TokenType == JsonTokenType.Null)
|
||||||
|
{
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Token is empty string.
|
||||||
if (reader.TokenType == JsonTokenType.String && ((reader.HasValueSequence && reader.ValueSequence.IsEmpty) || reader.ValueSpan.IsEmpty))
|
if (reader.TokenType == JsonTokenType.String && ((reader.HasValueSequence && reader.ValueSequence.IsEmpty) || reader.ValueSpan.IsEmpty))
|
||||||
{
|
{
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
return _baseJsonConverter.Read(ref reader, typeToConvert, options);
|
return JsonSerializer.Deserialize<TStruct>(ref reader, options);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <inheritdoc />
|
/// <inheritdoc />
|
||||||
public override void Write(Utf8JsonWriter writer, T? value, JsonSerializerOptions options)
|
public override void Write(Utf8JsonWriter writer, TStruct? value, JsonSerializerOptions options)
|
||||||
{
|
{
|
||||||
_baseJsonConverter.Write(writer, value, options);
|
if (value.HasValue)
|
||||||
|
{
|
||||||
|
JsonSerializer.Serialize(writer, value.Value, options);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
writer.WriteNullValue();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,27 @@
|
||||||
|
using System;
|
||||||
|
using System.Text.Json;
|
||||||
|
using System.Text.Json.Serialization;
|
||||||
|
|
||||||
|
namespace MediaBrowser.Common.Json.Converters
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Json nullable struct converter factory.
|
||||||
|
/// </summary>
|
||||||
|
public class JsonNullableStructConverterFactory : JsonConverterFactory
|
||||||
|
{
|
||||||
|
/// <inheritdoc />
|
||||||
|
public override bool CanConvert(Type typeToConvert)
|
||||||
|
{
|
||||||
|
return typeToConvert.IsGenericType
|
||||||
|
&& typeToConvert.GetGenericTypeDefinition() == typeof(Nullable<>)
|
||||||
|
&& typeToConvert.GenericTypeArguments[0].IsValueType;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <inheritdoc />
|
||||||
|
public override JsonConverter CreateConverter(Type typeToConvert, JsonSerializerOptions options)
|
||||||
|
{
|
||||||
|
var structType = typeToConvert.GenericTypeArguments[0];
|
||||||
|
return (JsonConverter)Activator.CreateInstance(typeof(JsonNullableStructConverter<>).MakeGenericType(structType));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -39,14 +39,9 @@ namespace MediaBrowser.Common.Json
|
||||||
NumberHandling = JsonNumberHandling.AllowReadingFromString
|
NumberHandling = JsonNumberHandling.AllowReadingFromString
|
||||||
};
|
};
|
||||||
|
|
||||||
// Get built-in converters for fallback converting.
|
|
||||||
var baseNullableInt32Converter = (JsonConverter<int?>)options.GetConverter(typeof(int?));
|
|
||||||
var baseNullableInt64Converter = (JsonConverter<long?>)options.GetConverter(typeof(long?));
|
|
||||||
|
|
||||||
options.Converters.Add(new JsonGuidConverter());
|
options.Converters.Add(new JsonGuidConverter());
|
||||||
options.Converters.Add(new JsonStringEnumConverter());
|
options.Converters.Add(new JsonStringEnumConverter());
|
||||||
options.Converters.Add(new JsonNullableStructConverter<int>(baseNullableInt32Converter));
|
options.Converters.Add(new JsonNullableStructConverterFactory());
|
||||||
options.Converters.Add(new JsonNullableStructConverter<long>(baseNullableInt64Converter));
|
|
||||||
|
|
||||||
return options;
|
return options;
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue
Block a user