Сериализиране на типове стойности с json.net

Опитвам се да направя ajax повикване, за да получа щатите от държава, но продължавам да получавам изключение за invalidcast. Използвам MVC4 с json.NET. Мога да сериализирам обектите без проблем в моите тестове, но когато направя извикването на ajax, продължавам да получавам грешка.

Това е типът стойност:

[JsonConverter(typeof(ValueObjectConverter))]
public class Code : IValueObject
{
    private readonly string _code;

    private Code(string code)
    {
        _code = code;
    }

    public override string ToString()
    {
        return _code;
    }

    public static implicit operator Code(string code)
    {
        return new Code(code);
    }

    public static implicit operator String(Code code)
    {
        return code.ToString();
    }
}

Това е моят обект:

public class CountryState : Entity
{
    public Code CodeWPS { get; set; }
    public Code CodeIAM { get; set; }
    public Name Description { get; set; }
}

Това е jsonconverter:

public class ValueObjectConverter : JsonConverter
{
    public override bool CanConvert(Type objectType)
    {
        return (objectType == typeof(IValueObject));
    }

    public override object ReadJson(JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer)
    {
        return (IValueObject)existingValue;
    }

    public override void WriteJson(JsonWriter writer, object value, JsonSerializer serializer)
    {
        var item = (IValueObject)value;
        writer.WriteValue(item.ToString());
        writer.Flush();
    }
}

Това е API методът:

    public IEnumerable<CountryState> GetStatesByCountry(string codeType, string countryCodeIam)
    {
        var states = _getAllCountriesQueryHandler.Handle(countryCodeIam);
        return states;
    }

Това е грешката, която получавам с firebug:

    {"Message":"An error has occurred.","ExceptionMessage":"The 'ObjectContent`1' type    failed to serialize the response body for content type 'application/json; charset=utf-8'.","ExceptionType":"System.InvalidOperationException","StackTrace":null,"InnerException":{"Message":"An error has occurred.","ExceptionMessage":"Error getting value from 'CodeWPS' on 'CNH.CSCN.BBS.Entities.CountryState'.","ExceptionType":"Newtonsoft.Json.JsonSerializationException","StackTrace":"   at Newtonsoft.Json.Serialization.DynamicValueProvider.GetValue(Object target)\r\n   at Newtonsoft.Json.Serialization.JsonSerializerInternalWriter.CalculatePropertyValues(JsonWriter writer, Object value, JsonContainerContract contract, JsonProperty member, JsonProperty property, JsonContract& memberContract, Object& memberValue)\r\n   at Newtonsoft.Json.Serialization.JsonSerializerInternalWriter.SerializeObject(JsonWriter writer, Object value, JsonObjectContract contract, JsonProperty member, JsonContainerContract collectionContract, JsonProperty containerProperty)\r\n   at Newtonsoft.Json.Serialization.JsonSerializerInternalWriter.SerializeList(JsonWriter writer, IEnumerable values, JsonArrayContract contract, JsonProperty member, JsonContainerContract collectionContract, JsonProperty containerProperty)\r\n   at Newtonsoft.Json.Serialization.JsonSerializerInternalWriter.Serialize(JsonWriter jsonWriter, Object value, Type objectType)\r\n   at Newtonsoft.Json.JsonSerializer.SerializeInternal(JsonWriter jsonWriter, Object value, Type objectType)\r\n   at System.Net.Http.Formatting.JsonMediaTypeFormatter.<>c__DisplayClassd.<WriteToStreamAsync>b__c()\r\n   at System.Threading.Tasks.TaskHelpers.RunSynchronously(Action action, CancellationToken token)","InnerException":{"Message":"An error has occurred.","ExceptionMessage":"Unable to cast object of type 'System.String' to type 'CNH.CSCN.BBS.Entities.ValueTypes.Code'.","ExceptionType":"System.InvalidCastException","StackTrace":"   at GetCodeWPS(Object )\r\n   at Newtonsoft.Json.Serialization.DynamicValueProvider.GetValue(Object target)"}}}

АКТУАЛИЗАЦИЯ:

Добавих преобразувател на договори:

public class SpecialContractResolver : DefaultContractResolver
{
    protected override IValueProvider CreateMemberValueProvider(MemberInfo member)
    {
        if (member.MemberType == MemberTypes.Property)
        {
            var pi = (PropertyInfo)member;
            if (typeof(IValueObject).IsAssignableFrom(pi.PropertyType))
            {
                return new IValueObjectValueProvider(member, pi.PropertyType);
            }
        }
        else if (member.MemberType == MemberTypes.Field)
        {
            var fi = (FieldInfo)member;
            if (typeof(IValueObject).IsAssignableFrom(fi.FieldType))
            {
                return new IValueObjectValueProvider(member, fi.FieldType);
            }
        }

        return base.CreateMemberValueProvider(member);
    }
}

и IValueProvider:

public class IValueObjectValueProvider : IValueProvider
{
    private readonly object _defaultValue;
    private readonly IValueProvider _underlyingValueProvider;


    public IValueObjectValueProvider(MemberInfo memberInfo, Type underlyingType)
    {
        _underlyingValueProvider = new DynamicValueProvider(memberInfo);
        _defaultValue = underlyingType;
    }

    public void SetValue(object target, object value)
    {
        target = value;
    }

    public object GetValue(object target)
    {
        return target.ToString();
    }
}

Знам, че съм на прав път с това, вече не получавам съобщение за грешка, но този резултат:

Обект { CodeWPS="CNH.CSCN.BBS.Entities.CountryState", CodeIAM="CNH.CSCN.BBS.Entities.CountryState", Description="CNH.CSCN.BBS.Entities.CountryState"}

Мисля, че моят contractresolver и ivalueprovider все още не са 100% правилни...


person Selketjah    schedule 09.08.2013    source източник
comment
След допълнително разследване изглежда, че той не използва моя персонализиран конвертор, когато прави ajax повикване.   -  person Selketjah    schedule 09.08.2013


Отговори (2)


Това е заради тази част от кода:

public override bool CanConvert(Type objectType)
{
    return (objectType == typeof(IValueObject));
}

Вашето objectType не е IValueObject, а Code - така че CanConvert връща false и вашият конвертор не се използва. Вашият код трябва да изглежда така:

return typeof(IValueObject).IsAssignableFrom(objectType);

С това условие се съпоставят и изпълняващите класове.

person fero    schedule 12.08.2013
comment
Благодаря ви за съвета, за съжаление това не разрешава моето невалидно изключение за предаване. CanConvert получава тези входни данни: въведете списък - въведете countrystate - въведете httperror - въведете низ... - person Selketjah; 12.08.2013
comment
Получавате грешка при кастинг, защото се опитвате да разопаковате existingValue до IValueObject, което не е възможно, защото все още не е десериализирано. Първо трябва да десериализирате JSON в C# низ и след това да го конвертирате в Code обект, което е доста сложно, защото имате само имплицитния оператор за преобразуване на типа, за да създадете нов Code обект. Ако вашият конструктор беше публичен, можете да напишете нещо като return Activator.CreateInstance(objectType, serializer.Deserialize<string>(reader)); или можете да използвате друго претоварване на CreateInstance, за да използвате частен конструктор. - person fero; 12.08.2013
comment
Той никога не достига до моите методи ReadJson или WriteJson, в момента разглеждам DynamicValueProvider на JSON.NET, за да видя как работи... - person Selketjah; 13.08.2013

Създадох SpecialContractResolver

public class SpecialContractResolver : DefaultContractResolver
{
    protected override IValueProvider CreateMemberValueProvider(MemberInfo member)
    {
        if (member.MemberType == MemberTypes.Property)
        {
            var pi = (PropertyInfo)member;
            if (typeof(IValueObject).IsAssignableFrom(pi.PropertyType))
            {
                return new ExpressionValueProvider(member);
            }
        }

        return base.CreateMemberValueProvider(member);
    }
}

Създадох ValueObjectConverter:

public class ValueObjectConverter : JsonConverter
{
    public override bool CanConvert(Type objectType)
    {
        return typeof(IValueObject).IsAssignableFrom(objectType);
    }

    public override object ReadJson(JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer)
    {
        return Activator.CreateInstance(objectType, serializer.Deserialize<string>(reader));
    }

    public override void WriteJson(JsonWriter writer, object value, JsonSerializer serializer)
    {
        var item = (IValueObject)value;
        writer.WriteValue(item.ToString());
        writer.Flush();
    }
}

И го добавих към моя WebApiConfig:

        var jsonFormatter = config.Formatters.JsonFormatter;
        jsonFormatter.SerializerSettings.ContractResolver = new SpecialContractResolver();
        jsonFormatter.SerializerSettings.Converters.Insert(0, new ValueObjectConverter());

Надявам се това да помогне на някого!

person Selketjah    schedule 13.08.2013