Instance validation error: * is not a valid value for *
Asked Answered
K

5

6

I'm trying to deserialize an XML string, where the value of an element, ain't within the scope of my Enum values.

Public enum MyEnum
{
    Unknown,
    Car,
    Bicycle,
    Boat
}

[SerializableAttribute()]
public class MyClass
{
    private string _id;
    private MyEnum _myEnum;

    public string ID
    {
        get { return _id; }
        set { _id = value; }
    }

    public MyEnum EnumValue
    {
        get { return _myEnum; }
        set { _myEnum = value; }
    }

    public MyClass(string id)
    {
        this._id = id;
    }

    public MyClass() : this("") { }
}

If I try to deserialize following string (note Plane as enum value):

<?xml version=\"1.0\" encoding=\"UTF-8\" standalone=\"no\"?><MyClass><ID>1234567890123456789</ID><EnumValue>Plane</EnumValue></MyClass>

then my deserialize will thrown an exception, before it even hit my public field for EnumValue, with following exception message:

Instance validation error: 'Plane' is not a valid value for EnumValue

Is it possible to return a default value for EnumValue, if the value I try to parse in the XML ain't supported as a EnumValue?? Eg. in the case of the XML string provided here, the EnumValue should be set as 'Unknown'.

Koziel answered 7/3, 2014 at 8:57 Comment(0)
T
6
[XmlIgnore]
public MyEnum EnumValueReal
{
    get { return _myEnum; }
    set { _myEnum = value; }
}

public string EnumValue
{
     get
     {
         return EnumValueReal.ToString();
     }

     set
     {
         MyEnum result = MyEnum.Unknown;
         Enum.TryParse(value, true, out result);

         EnumValueReal = result;
     }
}
Terryn answered 7/3, 2014 at 9:7 Comment(3)
EnumValue will be serialized and it allow to keep values that violate the MyEnum restriction. the same time the EnumValueReal can handle the constraint violation situation. Wasn't that the question that was asked?Terryn
Sorry. my bad. I just saw XmlIgore mentioned in EnumValueReal. didn't look at EnumValue. Sorry again. (+1)Havana
Ended with this solution, but without creating a public MyEnum field with XmlIgnore tag on, since the private MyEnum myEnum in MyClass.cs won't be serialized anyways. Converting the Enum to a String, was the solution in my case. Thanks.Koziel
P
1

Other way around would be to declare EnumValue as string and parse value in EnumValue property to MyEnum in another property (with custom logic). Another property should be marked as not serializable.

public string EnumValue
{
    get { return _myEnum; }
    set { _myEnum = value; }
}

[NonSerialized]
public MyEnum EnumValueTyped {
  get {
    MyEnum value;
    if (Enum.TryParse<MyEnum>(EnumValue, out value)) {
      return value;
    }
    return MyEnum.Unknown;
  }
  set {
    EnumValue = value.ToString();
  }
}
Profligate answered 7/3, 2014 at 9:6 Comment(0)
O
1

Sometimes what happens is we do not take the update of the dlls or the projects we are referring in the project after making changes to the later and hence the parameter added/deleted does not get detected, thus throwing the same issue. Hence better take the updated dll and proceed.Can be a silly mistake but often committed. All the best :)

Onanism answered 10/9, 2015 at 13:26 Comment(0)
I
1

IMHO the most 'appropriate' solution would be to make EnumValue property nullable - since error you are getting (and MyEnum.Unknown) implies that it is possible for this property not to have a value...

Code would look following:

public enum MyEnum
{
    Car,
    Bicycle,
    Boat
}

[Serializable()]
public class MyClass
{
    private string _id;
    private MyEnum? _myEnum;

    public string ID
    {
        get { return _id; }
        set { _id = value; }
    }

    public MyEnum? EnumValue
    {

        get { return _myEnum; }
        set { _myEnum = value; }
    }

    public MyClass(string id)
    {
        this._id = id;
    }

    public MyClass() : this("")
    {
    }
}
Interminable answered 27/9, 2016 at 10:29 Comment(1)
That doesn't quite address the problem, no.Maybe
F
0

Yes, but you can't do it anymore with SerializableAttribute I think.

You should implement ISerializable and provide your own serializer/deserializer. You can use the default serializer (new BinaryFormatter().serializer() e.g), but you have to implement a custom deserialization.

Fite answered 7/3, 2014 at 9:4 Comment(0)

© 2022 - 2024 — McMap. All rights reserved.