如何将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
是内部的,因此只能在包内调用。
我想通过存储过程将 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
是内部的,因此只能在包内调用。