ओपी के आधार पर विस्तृत रूप से टिप्पणियों में बताया गया है , एक अधिक उपयुक्त समाधान मौजूद है। ओपी बताता है कि वह अपने लॉग में कस्टम डेटा को लॉग 4नेट, अनुरोधों से संबंधित डेटा के साथ जोड़ना चाहता है।
प्रत्येक लॉग 4net कॉल को एक कस्टम सेंट्रलाइज्ड लॉग कॉल में लपेटने के बजाय, जो रिलेटेड रिक्वेस्ट रिलेटेड डेटा (प्रत्येक लॉग कॉल पर) को हैंडल करता है, लॉग में 4 अतिरिक्त कस्टम डेटा को सेट करने के लिए log4net फीचर्स डिक्शनरी डिक्लेयर करता है। उन dictionnaries का उपयोग करके आपके अनुरोध लॉग डेटा को शुरुआती अनुरोध पर वर्तमान अनुरोध के लिए रखने की अनुमति देता है, फिर उसे EndRequest इवेंट में खारिज करने के लिए। बीच में कोई भी लॉग उन कस्टम डेटा से लाभान्वित होगा।
और जो चीजें अनुरोध के संदर्भ में नहीं होती हैं, वे अनुरोध उपलब्धता के लिए परीक्षण करने की आवश्यकता को समाप्त करते हुए अनुरोध संबंधी डेटा लॉग करने का प्रयास नहीं करेंगे। यह समाधान उस सिद्धांत से मेल खाता है जो अरमान मैकहिट्रान अपने उत्तर में सुझा रहा था ।
काम करने के लिए इस समाधान के लिए, आपको अपने कस्टम डेटा को लॉग करने के लिए अपने log4net ऐपेंडर्स पर कुछ अतिरिक्त कॉन्फ़िगरेशन की भी आवश्यकता होगी।
यह समाधान कस्टम लॉग एन्हांसमेंट मॉड्यूल के रूप में आसानी से लागू किया जा सकता है। यहाँ इसके लिए नमूना कोड है:
using System;
using System.Web;
using log4net;
using log4net.Core;
namespace YourNameSpace
{
public class LogHttpModule : IHttpModule
{
public void Dispose()
{
// nothing to free
}
private const string _ipKey = "IP";
private const string _urlKey = "URL";
private const string _refererKey = "Referer";
private const string _userAgentKey = "UserAgent";
private const string _userNameKey = "userName";
public void Init(HttpApplication context)
{
context.BeginRequest += WebAppli_BeginRequest;
context.PostAuthenticateRequest += WebAppli_PostAuthenticateRequest;
// All custom properties must be initialized, otherwise log4net will not get
// them from HttpContext.
InitValueProviders(_ipKey, _urlKey, _refererKey, _userAgentKey,
_userNameKey);
}
private void InitValueProviders(params string[] valueKeys)
{
if (valueKeys == null)
return;
foreach(var key in valueKeys)
{
GlobalContext.Properties[key] = new HttpContextValueProvider(key);
}
}
private void WebAppli_BeginRequest(object sender, EventArgs e)
{
var currContext = HttpContext.Current;
currContext.Items[_ipKey] = currContext.Request.UserHostAddress;
currContext.Items[_urlKey] = currContext.Request.Url.AbsoluteUri;
currContext.Items[_refererKey] = currContext.Request.UrlReferrer != null ?
currContext.Request.UrlReferrer.AbsoluteUri : null;
currContext.Items[_userAgentKey] = currContext.Request.UserAgent;
}
private void WebAppli_PostAuthenticateRequest(object sender, EventArgs e)
{
var currContext = HttpContext.Current;
// log4net doc states that %identity is "extremely slow":
// http://logging.apache.org/log4net/release/sdk/log4net.Layout.PatternLayout.html
// So here is some custom retrieval logic for it, so bad, especialy since I
// tend to think this is a missed copy/paste in that documentation.
// Indeed, we can find by inspection in default properties fetch by log4net a
// log4net:Identity property with the data, but it looks undocumented...
currContext.Items[_userNameKey] = currContext.User.Identity.Name;
}
}
// General idea coming from
// http://piers7.blogspot.fr/2005/12/log4net-context-problems-with-aspnet.html
// We can not use log4net ThreadContext or LogicalThreadContext with asp.net, since
// asp.net may switch thread while serving a request, and reset the call context
// in the process.
public class HttpContextValueProvider : IFixingRequired
{
private string _contextKey;
public HttpContextValueProvider(string contextKey)
{
_contextKey = contextKey;
}
public override string ToString()
{
var currContext = HttpContext.Current;
if (currContext == null)
return null;
var value = currContext.Items[_contextKey];
if (value == null)
return null;
return value.ToString();
}
object IFixingRequired.GetFixedObject()
{
return ToString();
}
}
}
इसे अपनी साइट पर जोड़ें, IIS 7+ conf नमूना:
<system.webServer>
<!-- other stuff removed ... -->
<modules>
<!-- other stuff removed ... -->
<add name="LogEnhancer" type="YourNameSpace.LogHttpModule, YourAssemblyName" preCondition="managedHandler" />
<!-- other stuff removed ... -->
</modules>
<!-- other stuff removed ... -->
</system.webServer>
और उन अतिरिक्त गुणों, नमूना विन्यास को लॉग करने के लिए परिशिष्ट स्थापित करें:
<log4net>
<appender name="RollingLogFileAppender" type="log4net.Appender.RollingFileAppender">
<!-- other stuff removed ... -->
<layout type="log4net.Layout.PatternLayout">
<conversionPattern value="%date [%thread] %-5level %logger - %message - %property%newline%exception" />
</layout>
</appender>
<appender name="SqlAppender" type="log4net.Appender.AdoNetAppender">
<!-- other stuff removed ... -->
<commandText value="INSERT INTO YourLogTable ([Date],[Thread],[Level],[Logger],[UserName],[Message],[Exception],[Ip],[Url],[Referer],[UserAgent]) VALUES (@log_date, @thread, @log_level, @logger, @userName, @message, @exception, @Ip, @Url, @Referer, @UserAgent)" />
<!-- other parameters removed ... -->
<parameter>
<parameterName value="@userName" />
<dbType value="String" />
<size value="255" />
<layout type="log4net.Layout.PatternLayout">
<conversionPattern value="%property{userName}" />
</layout>
</parameter>
<parameter>
<parameterName value="@Ip"/>
<dbType value="String" />
<size value="255" />
<layout type="log4net.Layout.PatternLayout">
<conversionPattern value="%property{Ip}" />
</layout>
</parameter>
<parameter>
<parameterName value="@Url"/>
<dbType value="String" />
<size value="255" />
<layout type="log4net.Layout.PatternLayout">
<conversionPattern value="%property{Url}" />
</layout>
</parameter>
<parameter>
<parameterName value="@Referer"/>
<dbType value="String" />
<size value="255" />
<layout type="log4net.Layout.PatternLayout">
<conversionPattern value="%property{Referer}" />
</layout>
</parameter>
<parameter>
<parameterName value="@UserAgent"/>
<dbType value="String" />
<size value="255" />
<layout type="log4net.Layout.PatternLayout">
<conversionPattern value="%property{UserAgent}" />
</layout>
</parameter>
</appender>
<!-- other stuff removed ... -->
</log4net>