3
votes

Comment désérialiser le XML qui utilise une clé dans un objet

J'ai un projet MVC ASP.NET core 2.1 et je récupère des données à partir d'un récepteur Serlilog MSSqlServr qui stocke des valeurs dans un champ Propriétés en tant que type de données XML. Je souhaite désérialiser ces données dans un modèle de vue afin de pouvoir présenter les éléments individuels sous forme de données dans une vue.

Voici un exemple du XML du champ Propriétés de la table Logs de la base de données.

<div class="card-body ml3 mr3">


    @foreach (var logProperty in Model.XmlProperties.Property)
    {
        <div class="row">
            <div class="col-4 bg-light border border-primary">
                <span class="font-weight-bold">@logProperty.Key</span>
            </div>
            <div class="col-8 bg-secondary border border-left-0 border-primary">
                <span>@logProperty.Value</span>
            </div>
        </div>


    }

</div>

J'ai configuré une classe pour le décodage comme suit;

public class DetailLogWithPropertiesViewModel
{
    public int Id { get; set; }

    [Display(Name = "Message")]
    public string Message { get; set; }

    [Display(Name = "Level")]

    public string Level { get; set; }

    [Display(Name = "Time Stamp")]
    public DateTimeOffset TimeStamp { get; set; }

    [Display(Name = "Exception")]
    public string Exception { get; set; }

    [Display(Name = "Properties")]
    public string Properties { get; set; }

    public LogProperties XmlProperties { get; set; }

}

Et dans mon contrôleur, j'ai le code suivant;

        var response = await _client.GetLogAsync(id, $"api/logs", token);
        if (response == null)
        {
            return NotFound($"Unable to find a record for Log ID [{id}].");
        }

        var log = _mapper.Map<DetailLogViewModel>(response.Record);

        var serializer = new XmlSerializer(typeof(LogProperties));

        LogProperties logProperties;

        using (TextReader reader = new StringReader(log.Properties))
        {
            logProperties = (LogProperties)serializer.Deserialize(reader);
        }

        var logWithProperties = new DetailLogWithPropertiesViewModel
        {
            Id = log.Id,
            Message = log.Message,
            TimeStamp = log.TimeStamp,
            Exception = log.Exception,
            XmlProperties = logProperties 
        };


        return View(logWithProperties);

Mais la variable logProperties ne contient rien, donc je suppose que je ont les attributs XML incorrects dans la classe LogProperties.

J'ai passé pas mal de temps à chercher une solution et j'ai passé en revue tous les articles associés en entrant cette question mais je ne suis pas en mesure de trouver un exemple où le XML utilise "property key =" ou comment gérer une propriété d'attribut "key =" (si c'est le terme correct)

Des idées?

[MISE À JOUR 21/02/19]

J'ai fini par utiliser la suggestion @jdweng car c'était la moins complexe et elle m'a donné exactement ce que je voulais.

J'ai créé 2 classes (car j'aime garder mes fichiers de classe séparés en tant que préférences personnelles). Les classes sont ci-dessous:

using System.Xml.Serialization;

namespace PS.Models.ApiLogs
{
    [XmlRoot("property")]
    public class LogProperty
    {
        [XmlAttribute("key")]
        public string Key { get; set; }
        [XmlText]
        public string Value { get; set; }
    }
}

et

using System.Collections.Generic;
using System.Xml.Serialization;

namespace PS.Models.ApiLogs
{
    [XmlRoot("properties")]
    public class LogProperties
    {
        [XmlElement("property")]
        public List<LogProperty> Property { get; set; }

    }
}

Ensuite, dans mon contrôleur, j'ai ce qui suit pour la méthode Detail;

        var serializer = new XmlSerializer(typeof(LogProperties));

        LogProperties logProperties;

        using (TextReader reader = new StringReader(log.Properties))
        {
            logProperties = (LogProperties)serializer.Deserialize(reader);
        }

Mon DetailLogWithPropertiesViewModel est ci-dessous;

using System.Xml.Serialization;

namespace PS.Models.ApiLogs
{
    [XmlRoot("properties")]
    public class LogProperties
    {

        [XmlElement("SourceContext")]
        public string SourceContext { get; set; }

        [XmlElement("ActionId")]
        public string ActionId { get; set; }

        [XmlElement("ActionName")]
        public string ActionName { get; set; }

        [XmlElement("RequestId")]
        public string RequestId { get; set; }

        [XmlElement("RequestPath")]
        public string RequestPath { get; set; }

        [XmlElement("CorrelationId")]
        public string CorrelationId { get; set; }

        [XmlElement("ConnectionId")]
        public string ConnectionId { get; set; }

        [XmlElement("MachineName")]
        public string MachineName { get; set; }

        [XmlElement("ThreadId")]
        public string ThreadId { get; set; }

    }
}

Et la partie pertinente de mon Detail.cshtml est ci-dessous; p >

<properties>
  <property key="EventId">
    <structure type="">
      <property key="Id">404</property>
    </structure>
  </property>
  <property key="ActionId">0592d9e8-f4fd-459f-96b3-2b787d01a754</property>
  <property key="ActionName">API.Controllers.CompletionsController.GetCompletion (PS.API)</property>
  <property key="RequestId">0HLJ2IL5A9:00000001</property>
  <property key="RequestPath">/api/completions/0</property>
  <property key="CorrelationId" />
  <property key="ConnectionId">0HLJ2IL59</property>
  <property key="MachineName">RD0003FF1</property>
  <property key="ThreadId">117</property>
</properties>

Le XML qui est généré par Serilog et stocké dans la base de données MS SQL a un nombre variable de propriétés, que je comprends maintenant sont représentées sous forme de paires clé / valeur. Cette méthode me permet donc de m'assurer que toutes les propriétés fournies sont affichées dans la visionneuse de journal du site Web.


1 commentaires

J'ai l'autre direction! Si vous essayez d'utiliser votre classe pour créer un fichier XML (ou une chaîne avec XML) à l'aide de XmlSerializer.Serialize (), vous verrez quel type de document XML vous avez décrit.


3 Réponses :


1
votes

Essayez de suivre:

    [XmlRoot("properties")]
    public class LogProperties
    {

        [XmlElement("property")]
        public List<LogProperty> property { get; set; }

    }
    [XmlRoot("property")]
    public class LogProperty
    {
        [XmlAttribute("key")]
        public string key { get; set; }
        [XmlText]
        public string value { get; set; }
    }


0 commentaires

0
votes

1) Copiez votre XML dans le presse-papiers ...
2) ... Ouvrez Visual Studio et créez un fichier cs vide ...
3) ... allez dans EDIT> Collage spécial> XML aux classes ( peut nécessiter l'installation de développement Web ASP.NET ) ...
3) ... donnera ce code:

/// <remarks/>
[System.SerializableAttribute()]
[System.ComponentModel.DesignerCategoryAttribute("code")]
[System.Xml.Serialization.XmlTypeAttribute(AnonymousType = true)]
[System.Xml.Serialization.XmlRootAttribute(Namespace = "", IsNullable = false)]
public partial class properties
{

    private propertiesProperty[] propertyField;

    /// <remarks/>
    [System.Xml.Serialization.XmlElementAttribute("property")]
    public propertiesProperty[] property
    {
        get
        {
            return this.propertyField;
        }
        set
        {
            this.propertyField = value;
        }
    }
}

/// <remarks/>
[System.SerializableAttribute()]
[System.ComponentModel.DesignerCategoryAttribute("code")]
[System.Xml.Serialization.XmlTypeAttribute(AnonymousType = true)]
public partial class propertiesProperty
{

    private propertiesPropertyStructure structureField;

    private string[] textField;

    private string keyField;

    /// <remarks/>
    public propertiesPropertyStructure structure
    {
        get
        {
            return this.structureField;
        }
        set
        {
            this.structureField = value;
        }
    }

    /// <remarks/>
    [System.Xml.Serialization.XmlTextAttribute()]
    public string[] Text
    {
        get
        {
            return this.textField;
        }
        set
        {
            this.textField = value;
        }
    }

    /// <remarks/>
    [System.Xml.Serialization.XmlAttributeAttribute()]
    public string key
    {
        get
        {
            return this.keyField;
        }
        set
        {
            this.keyField = value;
        }
    }
}

/// <remarks/>
[System.SerializableAttribute()]
[System.ComponentModel.DesignerCategoryAttribute("code")]
[System.Xml.Serialization.XmlTypeAttribute(AnonymousType = true)]
public partial class propertiesPropertyStructure
{

    private propertiesPropertyStructureProperty propertyField;

    private string typeField;

    /// <remarks/>
    public propertiesPropertyStructureProperty property
    {
        get
        {
            return this.propertyField;
        }
        set
        {
            this.propertyField = value;
        }
    }

    /// <remarks/>
    [System.Xml.Serialization.XmlAttributeAttribute()]
    public string type
    {
        get
        {
            return this.typeField;
        }
        set
        {
            this.typeField = value;
        }
    }
}

/// <remarks/>
[System.SerializableAttribute()]
[System.ComponentModel.DesignerCategoryAttribute("code")]
[System.Xml.Serialization.XmlTypeAttribute(AnonymousType = true)]
public partial class propertiesPropertyStructureProperty
{

    private string keyField;

    private ushort valueField;

    /// <remarks/>
    [System.Xml.Serialization.XmlAttributeAttribute()]
    public string key
    {
        get
        {
            return this.keyField;
        }
        set
        {
            this.keyField = value;
        }
    }

    /// <remarks/>
    [System.Xml.Serialization.XmlTextAttribute()]
    public ushort Value
    {
        get
        {
            return this.valueField;
        }
        set
        {
            this.valueField = value;
        }
    }
}

4) Oui, beaucoup de code o_O. Utilisez un XmlDeserializer(typeof(properties))


0 commentaires

0
votes

Si vous créez un exemple d'instance de votre classe LogProperties et que vous la sérialisez comme ceci:

// Must escape all quotes !
string xmlSample = @"
<properties>
    <property key=""EventId"">
        <structure type = """">
            <property key=""Id"">404</property>
        </structure>
    </property>
    <property key=""ActionId""> 0592d9e8 - f4fd - 459f - 96b3 - 2b787d01a754</property>
    <property key=""ActionName""> API.Controllers.CompletionsController.GetCompletion(PS.API)</property>
    <property key=""RequestId""> 0HLJ2IL5A9: 00000001</property>
    <property key=""RequestPath"">/api/completions/0</property>
    <property key=""CorrelationId"" />
    <property key=""ConnectionId"">0HLJ2IL59</property>
    <property key=""MachineName"">RD0003FF1</property>
    <property key=""ThreadId"">117</property>
</properties>";

StringReader reader = new StringReader(xmlSample);
XDocument xdoc = XDocument.Parse(xmlSample);

LogProperties log = new LogProperties();
foreach (XElement prop in xdoc.Root.Elements())
{
    switch (prop.Attribute("key").Value)
    {
    case "ActionId" : log.ActionId = prop.Value; break;
    case "ActionName" : log.ActionName = prop.Value; break;

    // and so on

    }
}

Voici ce que vous obtenez:

<?xml version="1.0" encoding="utf-16"?>
<properties xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:xsd="http://www.w3.org/2001/XMLSchema">
  <SourceContext>MySourceContext</SourceContext>
  <ActionId>MyActionId</ActionId>
  <ActionName>MyActionName</ActionName>
</properties>

Donc, la classe que vous avez ne correspond pas très bien au XML que vous avez. Il est toujours utile d'essayer la sérialisation dans les deux sens.

Voici une façon de sérialiser le XML que vous avez dans la classe que vous avez (en séparant clairement votre modèle d'objet et votre format de persistance). Il utilise la classe XDocument de l'espace de noms System.Xml.Linq.

var serializer = new XmlSerializer(typeof(LogProperties));

LogProperties logProperties = new LogProperties() 
{ 
    SourceContext = "MySourceContext",
    ActionId = "MyActionId",
    ActionName = "MyActionName"
};

StringBuilder sb = new StringBuilder();
using (StringWriter writer = new StringWriter(sb))
{
    serializer.Serialize(writer, logProperties);
}

Console.WriteLine(sb.ToString());

Notez qu'avec cette approche:

  • vous n'avez pas besoin des attributs XML de votre classe
  • vous n'avez pas besoin de façonner votre classe pour l'adapter au XML (ou l'inverse)
  • vous pouvez ajouter votre propre initialisation personnalisée de la classe LogProperties (plus besoin d'un constructeur par défaut)
  • vous pouvez ajouter votre propre gestion des erreurs personnalisée


0 commentaires