You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
Sonarr/src/NzbDrone.Core/ImportLists/TolerantEnumConverter.cs

161 lines
4.9 KiB

using System;
using System.Collections.Generic;
using System.Linq;
using System.Reflection;
using System.Runtime.Serialization;
using Newtonsoft.Json;
namespace NzbDrone.Core.ImportLists
{
/* This class was copied from gubenkoved TolerantEnumConverter.cs file, which is available from github
* https://gist.github.com/gubenkoved/999eb73e227b7063a67a50401578c3a7
*/
public class TolerantEnumConverter : JsonConverter
{
[ThreadStatic]
private static Dictionary<Type, Dictionary<string, object>> _fromValueMap; // string representation to Enum value map
[ThreadStatic]
private static Dictionary<Type, Dictionary<object, string>> _toValueMap; // Enum value to string map
public string UnknownValue { get; set; } = "Unknown";
public override bool CanConvert(Type objectType)
{
var type = IsNullableType(objectType) ? Nullable.GetUnderlyingType(objectType) : objectType;
return type.IsEnum;
}
public override object ReadJson(JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer)
{
var isNullable = IsNullableType(objectType);
var enumType = isNullable ? Nullable.GetUnderlyingType(objectType) : objectType;
InitMap(enumType);
if (reader.TokenType == JsonToken.String)
{
var enumText = reader.Value.ToString();
var val = FromValue(enumType, enumText);
if (val != null)
{
return val;
}
}
else if (reader.TokenType == JsonToken.Integer)
{
var enumVal = Convert.ToInt32(reader.Value);
var values = (int[])Enum.GetValues(enumType);
if (values.Contains(enumVal))
{
return Enum.Parse(enumType, enumVal.ToString());
}
}
if (!isNullable)
{
var names = Enum.GetNames(enumType);
var unknownName = names
.Where(n => string.Equals(n, UnknownValue, StringComparison.OrdinalIgnoreCase))
.FirstOrDefault();
if (unknownName == null)
{
throw new JsonSerializationException($"Unable to parse '{reader.Value}' to enum {enumType}. Consider adding Unknown as fail-back value.");
}
return Enum.Parse(enumType, unknownName);
}
return null;
}
public override void WriteJson(JsonWriter writer, object value, JsonSerializer serializer)
{
var enumType = value.GetType();
InitMap(enumType);
var val = ToValue(enumType, value);
writer.WriteValue(val);
}
private bool IsNullableType(Type t)
{
return t.IsGenericType && t.GetGenericTypeDefinition() == typeof(Nullable<>);
}
private void InitMap(Type enumType)
{
if (_fromValueMap == null)
{
_fromValueMap = new Dictionary<Type, Dictionary<string, object>>();
}
if (_toValueMap == null)
{
_toValueMap = new Dictionary<Type, Dictionary<object, string>>();
}
if (_fromValueMap.ContainsKey(enumType))
{
return; // already initialized
}
var fromMap = new Dictionary<string, object>(StringComparer.InvariantCultureIgnoreCase);
var toMap = new Dictionary<object, string>();
var fields = enumType.GetFields(BindingFlags.Static | BindingFlags.Public);
foreach (var field in fields)
{
var name = field.Name;
var enumValue = Enum.Parse(enumType, name);
// use EnumMember attribute if exists
var enumMemberAttrbiute = field.GetCustomAttribute<EnumMemberAttribute>();
if (enumMemberAttrbiute != null)
{
var enumMemberValue = enumMemberAttrbiute.Value;
fromMap[enumMemberValue] = enumValue;
toMap[enumValue] = enumMemberValue;
}
else
{
toMap[enumValue] = name;
}
fromMap[name] = enumValue;
}
_fromValueMap[enumType] = fromMap;
_toValueMap[enumType] = toMap;
}
private string ToValue(Type enumType, object obj)
{
var map = _toValueMap[enumType];
return map[obj];
}
private object FromValue(Type enumType, string value)
{
var map = _fromValueMap[enumType];
if (!map.ContainsKey(value))
{
return null;
}
return map[value];
}
}
}