我在我编写的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来处理它?谢谢。
答案 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
目录。