I have a custom type:
[TypeConverter(typeof(FriendlyUrlTypeConverter))]
public class FriendlyUrl : IEquatable<FriendlyUrl>, IConvertible
{
public FriendlyUrl()
{
_friendlyUrl = string.Empty;
}
public FriendlyUrl(string value)
{
value = value.Trim();
if (!FriednlyUrlValidator.Validate(value))
throw new FriendlyUrlValidationException("Invalid value for FrienlyUrl");
_friendlyUrl = value;
}
public static implicit operator FriendlyUrl(string friendlyUrlValue)
{
if (friendlyUrlValue != "" && !FriednlyUrlValidator.Validate(friendlyUrlValue))
throw new FriendlyUrlValidationException($"Invalid value for FrienlyUrl: {friendlyUrlValue}");
return new FriendlyUrl { _friendlyUrl = friendlyUrlValue };
}
public static implicit operator string(FriendlyUrl furl)
{
return furl._friendlyUrl;
}
public override string ToString()
{
return _friendlyUrl;
}
private string _friendlyUrl;
//[...skip IEquatable implementation...]
TypeCode IConvertible.GetTypeCode()
{
return _friendlyUrl.GetTypeCode();
}
bool IConvertible.ToBoolean(IFormatProvider provider)
{
return ((IConvertible) _friendlyUrl).ToBoolean(provider);
}
byte IConvertible.ToByte(IFormatProvider provider)
{
return ((IConvertible) _friendlyUrl).ToByte(provider);
}
char IConvertible.ToChar(IFormatProvider provider)
{
return ((IConvertible) _friendlyUrl).ToChar(provider);
}
DateTime IConvertible.ToDateTime(IFormatProvider provider)
{
return ((IConvertible) _friendlyUrl).ToDateTime(provider);
}
decimal IConvertible.ToDecimal(IFormatProvider provider)
{
return ((IConvertible) _friendlyUrl).ToDecimal(provider);
}
double IConvertible.ToDouble(IFormatProvider provider)
{
return ((IConvertible) _friendlyUrl).ToDouble(provider);
}
short IConvertible.ToInt16(IFormatProvider provider)
{
return ((IConvertible) _friendlyUrl).ToInt16(provider);
}
int IConvertible.ToInt32(IFormatProvider provider)
{
return ((IConvertible) _friendlyUrl).ToInt32(provider);
}
long IConvertible.ToInt64(IFormatProvider provider)
{
return ((IConvertible) _friendlyUrl).ToInt64(provider);
}
sbyte IConvertible.ToSByte(IFormatProvider provider)
{
return ((IConvertible) _friendlyUrl).ToSByte(provider);
}
float IConvertible.ToSingle(IFormatProvider provider)
{
return ((IConvertible) _friendlyUrl).ToSingle(provider);
}
string IConvertible.ToString(IFormatProvider provider)
{
return _friendlyUrl.ToString(provider);
}
object IConvertible.ToType(Type conversionType, IFormatProvider provider)
{
if (conversionType == typeof(FriendlyUrl))
return this;
return ((IConvertible) _friendlyUrl).ToType(conversionType, provider);
}
ushort IConvertible.ToUInt16(IFormatProvider provider)
{
return ((IConvertible) _friendlyUrl).ToUInt16(provider);
}
uint IConvertible.ToUInt32(IFormatProvider provider)
{
return ((IConvertible) _friendlyUrl).ToUInt32(provider);
}
ulong IConvertible.ToUInt64(IFormatProvider provider)
{
return ((IConvertible) _friendlyUrl).ToUInt64(provider);
}
}
and here's the test for Json serialization/deserialization (xUnit):
[Fact]
public void ConvertToJsonAndBack()
{
FriendlyUrl friendlyUrl = "some-friendly-url-1";
string friendlyUrlJson = JsonConvert.SerializeObject(friendlyUrl);
Assert.Equal($"\"{friendlyUrl}\"", friendlyUrlJson);
// ******** Throws the next line: ********
FriendlyUrl deserialized = JsonConvert.DeserializeObject<FriendlyUrl>(friendlyUrlJson);
Assert.Equal(friendlyUrl, deserialized);
}
It throws with the exception:
Newtonsoft.Json.JsonSerializationException : Error converting value "some-friendly-url-1" to type 'BlahBlah.Entities.FriendlyUrl'. Path '', line 1, position 21. ---- System.InvalidCastException : Invalid cast from 'System.String' to 'BlahBlah.Entities.FriendlyUrl'.
Stack Trace:
Stack Trace: JsonSerializerInternalReader.EnsureType(JsonReader reader, Object value, CultureInfo culture, JsonContract contract, Type targetType) JsonSerializerInternalReader.CreateValueInternal(JsonReader reader, Type objectType, JsonContract contract, JsonProperty member, JsonContainerContract containerContract, JsonProperty containerMember, Object existingValue) JsonSerializerInternalReader.Deserialize(JsonReader reader, Type objectType, Boolean checkAdditionalContent) JsonSerializer.DeserializeInternal(JsonReader reader, Type objectType) JsonSerializer.Deserialize(JsonReader reader, Type objectType) JsonConvert.DeserializeObject(String value, Type type, JsonSerializerSettings settings) JsonConvert.DeserializeObject[T](String value, JsonSerializerSettings settings) JsonConvert.DeserializeObject[T](String value)
Now, it does work if I remove IConvertible implementation and leave only:
- explicit and implicit 'FriendlyUrl <=> string' conversion operators.
TypeConverterimplementation - see the very first line.
But when the class implements IConvertible I get that error. How can I fix this?
(I really need IConvertible to be implemented. I also tried to use JsonObject and it didn't help).
Fiddle
Here's the repro - https://dotnetfiddle.net/YPfr60.
TypeConverter
public class FriendlyUrlTypeConverter : TypeConverter
{
public override bool CanConvertFrom(ITypeDescriptorContext context, Type sourceType)
{
return sourceType == typeof(string) || base.CanConvertFrom(context, sourceType);
}
public override object ConvertFrom(ITypeDescriptorContext context, CultureInfo culture, object value)
{
return value is string sValue ? new FriendlyUrl(sValue) : base.ConvertFrom(context, culture, value);
}
public override object ConvertTo(ITypeDescriptorContext context, CultureInfo culture, object value, Type destinationType)
{
return destinationType == typeof(string) ? value.ToString() : base.ConvertTo(context, culture, value, destinationType);
}
}