Log4net xml appender不会创建根元素

时间:2016-01-26 10:26:22

标签: xml log4net element root appender

我在我编写的C#webservice中使用Log4net(v2.0.50727)。它配置为创建2个appender。第一个是我监控的一般调试文件,以确保服务正确运行,第二个是我以不同间隔转发给客户的报告(现在手动,但可能会自动化)。

<log4net>
  <appender name="RollingFileAppender" type="log4net.Appender.RollingFileAppender">
    <file value="..\logs\Service.log.xml"/>
    <appendToFile value="true"/>
    <rollingStyle value="Size"/>
    <maxSizeRollBackups value="20"/>
    <maximumFileSize value="10000KB"/>
    <staticLogFileName value="true"/>
    <preserveLogFileNameExtension value="true" />
    <layout type="log4net.Layout.XmlLayoutSchemaLog4j">
      <conversionPattern value="%utcdate [%thread] %-5level %logger [%property{NDC}] - %message%newline"/>
    </layout>
  </appender>
  <appender name="CustomerReportAppender" type="log4net.Appender.RollingFileAppender">
      <file value="..\logs\Customer.Report.xml"/>
      <appendToFile value="true"/>
      <rollingStyle value="Size"/>
      <maxSizeRollBackups value="20"/>
      <maximumFileSize value="10000KB"/>
      <staticLogFileName value="true"/>
      <preserveLogFileNameExtension value="true" />
      <layout type="log4net.Layout.XmlLayoutSchemaLog4j">
        <conversionPattern value="%utcdate [%thread] %-5level %logger [%property{NDC}] - %message%newline"/>
      </layout>
  </appender>
  <root>
    <level value="DEBUG"/>
    <appender-ref ref="RollingFileAppender"/>
  </root>
  <logger name="CustomerReport">
    <level value="ALL" />
    <appender-ref ref="CustomerReportAppender" />
  </logger>
</log4net>

对两个文件的记录都可以正常工作。问题是,如果我将Customer.Report.xml文件发送给我的客户,它将不会在Excel中打开,因为它是无效的XML。该文件由多行事件元素组成 -

<event logger="CustomerReport" timestamp="1453717399436" level="INFO" thread="8">
<message>Customer Info</message>
<properties>
<data name="UserName" value="IIS APPPOOL\Customer Service" />
<data name="log4jmachinename" value="MyServer" />
<data name="log4japp" value="/LM/W3SVC/3/ROOT-2-130981905203581113" />
<data name="HostName" value="MyServer" />
</properties>
</event>

我认为Excel正在抱怨,因为它没有1个根元素。如果我将整个文件包装在 -

<Report>
....
</Report>

然后Excel加载它就OK了。是否有可能让Log4net为我生成这个?或者可以Excel来处理它?谢谢。

1 个答案:

答案 0 :(得分:1)

PowerShell是一个很好的工具,可以快速解决这类问题。

不确定这是否是您正在寻找的答案类型,但它会以中间方式解决您的问题。您已经提到过,您现在手动执行此操作,但可能会在将来自动执行此操作 - PowerShell也是一种很好的方法。

因此,下面的脚本会要求您选择该文件,然后自动将其XML内容放在您指定的根元素中:

[CmdletBinding()]
Param(
    [Parameter(ValueFromPipeline=$true)]
    [ValidateScript({Test-Path $_ -PathType Container})]
    [string]
    $InitialDirectory,

    [ValidateNotNullOrEmpty()]
    [string]
    $RootElementName = 'Report'
)

[System.Reflection.Assembly]::LoadWithPartialName("System.windows.forms") | Out-Null

$OpenFileDialog = New-Object System.Windows.Forms.OpenFileDialog
$OpenFileDialog.InitialDirectory = $InitialDirectory
$OpenFileDialog.Filter = "XML (*.xml)| *.xml"
$OpenFileDialog.ShowDialog() | Out-Null

$FileName = $OpenFileDialog.FileName
$FileContent = Get-Content $FileName

$FileContentXml = [Xml]$FileContent

[XML.XMLDocument]$XMLDocument = New-Object System.XML.XMLDocument
[XML.XMLElement]$XMLRoot = $XMLDocument.CreateElement($RootElementName)
$XMLRoot.InnerXml = $FileContentXml.OuterXml
$XMLDocument.AppendChild($XMLRoot)

$XMLDocument.Save($FileName)

您可以通过将此脚本保存在新文件中来使用它,例如D:\Add-RootXmlElement.ps1,然后您可以从cmd提示符运行它,如下所示:

@powershell D:\Add-RootXmlElement.ps1 -InitialDirectory "D:\Logs" -RootElementName "Report"

根据需要调整Param块中的默认值,以便将来节省一些打字。

这会解决您的问题吗?

修改

当然,如果您想自动执行此操作并且在调用脚本时知道文件名,只需省略整个$InitialDirectory$OpenFileDialog部分,然后只需设置要添加根元素的日志文件路径的$FileName目录。