使用Log4Net AdoNetAppender记录大型消息

4

我有一个库,其中包含几个Web服务调用。现在的要求是记录所有对外部Web服务发出的请求以及记录接收到的所有响应。这是为了后续的内部审核。

我的SQL表格如下所示。

CREATE TABLE [dbo].[WebServiceLog](
    [Id] [int] IDENTITY(1,1) NOT NULL,
    [Date] [datetime] NOT NULL,
    [Thread] [nvarchar](255) NOT NULL,
    [Level] [nvarchar](50) NOT NULL,
    [Logger] [nvarchar](255) NOT NULL,
    [Message] [ntext] NOT NULL,
    [SessionID] [nvarchar](50) NOT NULL,
    [SearchID] [nvarchar](25) NULL
)

请注意,消息字段是存储来自Web服务的请求和响应的地方。有时响应非常大,超过90k个字符,甚至更多。(例如:航班/酒店可用性搜索-可能返回大量数据)
我的追加器配置如下。
<appender name="InfoAppender" type="log4net.Appender.AdoNetAppender">
        <bufferSize value="1"/>
        <connectionType value="System.Data.SqlClient.SqlConnection, System.Data, Version=2.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089"/>
        <connectionString value="Data Source=tcexpress;Initial Catalog=test;Persist Security Info=True;User ID=testUser;Password=**"/>
        <commandText value="INSERT INTO WebServiceLog ([Date],[Thread],[Level],[Logger],[Message],[SessionID], [SearchID]) VALUES (@log_date, @thread, @log_level, @logger, @message, @session_id, @search_id)"/>
        <parameter>
            <parameterName value="@log_date"/>
            <dbType value="DateTime"/>
            <layout type="log4net.Layout.RawTimeStampLayout"/>
        </parameter>
        <parameter>
            <parameterName value="@thread"/>
            <dbType value="String"/>
            <size value="255"/>
            <layout type="log4net.Layout.PatternLayout">
                <conversionPattern value="%thread"/>
            </layout>
        </parameter>
        <parameter>
            <parameterName value="@log_level"/>
            <dbType value="String"/>
            <size value="50"/>
            <layout type="log4net.Layout.PatternLayout">
                <conversionPattern value="%level"/>
            </layout>
        </parameter>
        <parameter>
            <parameterName value="@logger"/>
            <dbType value="String"/>
            <size value="255"/>
            <layout type="log4net.Layout.PatternLayout">
                <conversionPattern value="%logger"/>
            </layout>
        </parameter>
        <parameter>
            <parameterName value="@message"/>
            <dbType value="String"/>
            <size value="454751"/>
            <layout type="log4net.Layout.PatternLayout">
                <conversionPattern value="%message"/>
            </layout>
        </parameter>
        <parameter>
            <parameterName value="@session_id"/>
            <dbType value="String"/>
            <size value="50"/>
            <layout type="log4net.Layout.PatternLayout">
                <conversionPattern value="%X{sessionID}"/>
            </layout>
        </parameter>
        <parameter>
            <parameterName value="@search_id"/>
            <dbType value="String"/>
            <size value="25"/>
            <layout type="log4net.Layout.PatternLayout">
                <conversionPattern value="%X{searchID}"/>
            </layout>
        </parameter>
    </appender>

有没有办法告诉appender,我的字段是nText类型的,并且可以接受任意数量的数据?我尝试使用string设置一个非常大的字段长度值,但它似乎总是在454751处截断。
有人遇到过这个问题吗?或者是否有更好的方法使用log4net将大型消息记录到数据库中?任何帮助都将不胜感激。


6
你尝试过将size设置为-1吗?另外,454751看起来是一个很大的数字(40万?)。在将数据插入数据库之前最好先压缩数据。 - Dmitry Reznik
我尝试将大小设置为-1,但没有成功。当我这样做时,日志消息无法显示在数据库中(可能是由于内部错误)。 - stefann
将设置为-1对我无效。我的消息被截断为42k。 - Jacques
1个回答

7
也许这并不能完全回答你的问题,但如果你使用 nvarchar(max) 而不是 nText,那么这对你来说应该可以正常工作。
<parameter>
  <parameterName value="@message"/>
  <dbType value="String"/>
  <layout type="log4net.Layout.PatternLayout">
    <conversionPattern value="%message" />
  </layout>
</parameter>

这是我用来记录异常(带有堆栈跟踪)的方式。将其设置为-1也应该像Dmitriy建议的那样起作用。

4
将数据类型改为nvarchar(max) 对我没有起作用。然而,将大小设为-1 却非常有效。(使用SQL Server 2012后端) - batkuip
您IP地址为143.198.54.68,由于运营成本限制,当前对于免费用户的使用频率限制为每个IP每72小时10次对话,如需解除限制,请点击左下角设置图标按钮(手机用户先点击左上角菜单按钮)。 - Panagiotis Kanavos
1
这些答案还有效吗?我尝试了no size node和size value = -1,但我的列仍然被截断。 - Clint
我不知道这是否仍然有效,但是我当时建议的配置对我确实起作用了。显然,并不是每个人都有这种经历。 - Stefan Egli
1
自从最近的更新后,我必须使用-1来代替 nvarchar(max),否则我会收到 InvalidOperationException: SqlCommand.Prepare method requires all variable length parameters to have an explicitly set non-zero Size. 在此之前,不需要为 nvarchar(max) 指定大小就可以正常工作。 - Benjamin Freitag

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接