如何将eventLog.Properties转换成XML?

How to convert eventLog.Properties into XML?

我想通过存储过程将 Serilog 消息存储到数据库中。我创建了新的接收器。

    namespace Copacking.Web.Logging
    {
        class StoredProcedureSink : ILogEventSink
        {
            private readonly string connectionString;
            public StoredProcedureSink(string connectionString) {
                this.connectionString = connectionString;
            }

            public void Emit(LogEvent logEvent)
            {
var xml = new XElement("properties");

            foreach (KeyValuePair<string, LogEventPropertyValue> property in logEvent.Properties)
            {
                xml.Add(new XElement("property", new XAttribute("Key", property.Key), property.Value));
            }

                using (var conn = new SqlConnection(connectionString))
                using (var command = new SqlCommand("dbo.usp_insertLog", conn){ CommandType = CommandType.StoredProcedure })
                {
                    conn.Open();
                    command.Parameters.Add(new SqlParameter("@message", logEvent.RenderMessage()));
                    command.Parameters.Add(new SqlParameter("@messageTemplate", logEvent.MessageTemplate.Text));
                    command.Parameters.Add(new SqlParameter("@level", logEvent.Level.ToString()));
                    command.Parameters.Add(new SqlParameter("@timestamp", logEvent.Timestamp));
                    command.Parameters.Add(new SqlParameter("@exception", logEvent.Exception.ToString()));
                    command.Parameters.Add(new SqlParameter("@properties", xml.ToString()));
                    command.ExecuteNonQuery();
                }
            }

        }

    } 

它有效,但我得到

 <properties>
  <property Key="SourceContext">"..."</property>
  <property Key="RequestId">"..."</property>
  <property Key="RequestPath">"..."</property>
  <property Key="RequestHeader">[("Connection": "Keep-Alive"), ("Accept": "text/html, application/xhtml+xml, image/jxr, */*"), ("Accept-Encoding": "..."), ("Accept-Language": "..."), ("Host": "..."), ("User-Agent": "..."), ("MS-ASPNETCORE-TOKEN": "..."), ("MS-ASPNETCORE-WINAUTHTOKEN": "..."), ("X-Original-Proto": "..."), ("X-Original-For": "...")]</property>
  <property Key="ActivityId">"..."</property>
  <property Key="UserName">"..."</property>
</properties>

而不是

<properties>
  <property key="SourceContext">...</property>
  <property key="RequestId">...</property>
  <property key="RequestPath">...</property>
  <property key="RequestHeader">
    <dictionary>
      <item key="Connection">...</item>
      <item key="Accept">...</item>
      <item key="Accept-Encoding">...</item>
      <item key="Accept-Language">...</item>
      <item key="Host">...</item>
      <item key="User-Agent">...</item>
      <item key="MS-ASPNETCORE-TOKEN">...</item>
      <item key="MS-ASPNETCORE-WINAUTHTOKEN">...</item>
      <item key="X-Original-Proto">...</item>
      <item key="X-Original-For">...</item>
    </dictionary>
  </property>
  <property key="ActivityId">...</property>
  <property key="UserName">...</property>
</properties>

知道如何更改代码吗?也许实现一些递归?!

我用很少的解决方法解决了这个问题。我复制了以下 类

  • 列选项
  • 标准列
  • XmlPropertyFormatter

来自 serilog nuget 包 https://github.com/serilog/serilog-sinks-mssqlserver 并打电话给

command.Parameters.Add(new SqlParameter("@properties", ConvertPropertiesToXmlStructure(logEvent.Properties)));

我不能直接使用 nuget 包中的 类,因为 XmlPropertyFormatter 是内部的,因此只能在包内调用。