0

I'm converting an Object to json first, and then convert the json to xml. I need to do this to generate some of the properties as xml attribute instead of elements. Everything works as expected except I couldn't get it to generate seperate xml element for each item.

C# code:

string json = JsonConvert.SerializeObject(myObj);
XmlDocument xdoc = JsonConvert.DeserializeXmlNode(json, "root");

Generated json:

{
  "header": "myheader",
  "transaction": {
    "date": "2019-09-24",
    "items": [
      {
        "number": "123",
        "unit": "EA",
        "qty": 6
      },
      {
        "number": "456",
        "unit": "CS",
        "qty": 4
      }
    ]
  }
}

C# classes:

public class Item
{
    [JsonProperty("@number")]
    public string number { get; set; }
    [JsonProperty("@unit")]
    public string unit { get; set; }
    [JsonProperty("@qty")]
    public int qty { get; set; }
}

public class Transaction
{
    [JsonProperty("@date")]
    public string date { get; set; }
    public List<Item> items { get; set; }
}

public class Root
{
    public string header { get; set; }
    public Transaction transaction { get; set; }
}

Generated (not desired) output:

<root>
    <header>string</header>
    <transaction date="string">
        <items number="string" unit="string" qty="0"/>
        <items number="string" unit="string" qty="0"/>
    </transaction>
</root>

Expected output:

<root>
    <header>string</header>
    <transaction date="string">
        <items>
            <item number="string" unit="string" qty="0"/>
            <item number="string" unit="string" qty="0"/>
        </items>
    </transaction>
</root>
coder
  • 4,121
  • 14
  • 53
  • 88

1 Answers1

0

Followed the instructions on this link recommended by @Abhineet Chaudhary. It works as expected.

var jObject = JObject.Parse(json);
jObject.SelectToken("transaction").SelectTokens("items").WrapWithObjects("item");
var finalXml = jObject.ToXElement("root", false);

public static class JsonExtensions
{
    public static void WrapWithObjects(this IEnumerable<JToken> values, string name)
    {
        foreach (var value in values.ToList())
        {
            var newParent = new JObject();
            if (value.Parent != null)
                value.Replace(newParent);
            newParent[name] = value;
        }
    }

    public static XElement ToXElement(this JObject obj, string deserializeRootElementName = null, bool writeArrayAttribute = false)
    {
        if (obj == null)
            return null;
        using (var reader = obj.CreateReader())
            return JsonExtensions.DeserializeXElement(reader, deserializeRootElementName, writeArrayAttribute);
    }

    static XElement DeserializeXElement(JsonReader reader, string deserializeRootElementName, bool writeArrayAttribute)
    {
        var converter = new Newtonsoft.Json.Converters.XmlNodeConverter() { DeserializeRootElementName = deserializeRootElementName, WriteArrayAttribute = writeArrayAttribute };
        var jsonSerializer = JsonSerializer.CreateDefault(new JsonSerializerSettings { Converters = new JsonConverter[] { converter } });
        return jsonSerializer.Deserialize<XElement>(reader);
    }
}
coder
  • 4,121
  • 14
  • 53
  • 88