ওপির উপর ভিত্তি করে বিস্তারিত প্রয়োজনের মন্তব্যে ব্যাখ্যা করা হয়েছে , আরও একটি উপযুক্ত সমাধান বিদ্যমান। ওপিতে বলা হয়েছে যে সে লগ 4 নেট, অনুরোধ সম্পর্কিত ডেটা সহ এর লগগুলিতে কাস্টম ডেটা যুক্ত করতে চায়।
প্রতিটি লগ 4 নেট কলকে একটি কাস্টম সেন্ট্রালাইজড লগ কলটিতে মোড়কের পরিবর্তে অনুরোধ সম্পর্কিত ডেটা পুনরুদ্ধার পরিচালনা করে (প্রতিটি লগ কলে), লগ 4নেটে লগ-ইন করার জন্য কাস্টম অতিরিক্ত ডেটা সেট করার জন্য প্রসঙ্গের অভিধানের বৈশিষ্ট্য রয়েছে। এই ডিকশনারিগুলি ব্যবহার করে আপনার অনুরোধের লগ ডেটা বর্তমান অনুরোধের জন্য বিনিআরকোয়েস্ট ইভেন্টে স্থাপন করার অনুমতি দেয়, তারপরে এটিকে এন্ডরেকোয়েস্ট ইভেন্টে বরখাস্ত করে। এর মধ্যে যে কোনও লগ ইন সেই কাস্টম ডেটা থেকে উপকৃত হবে।
এবং অনুরোধের প্রসঙ্গে যে জিনিসগুলি ঘটে না সেগুলি অনুরোধের প্রাপ্যতার জন্য পরীক্ষার প্রয়োজনীয়তা বাদ দিয়ে অনুরোধ সম্পর্কিত ডেটা লগ করার চেষ্টা করবে না। এই সমাধানটি তার উত্তরে আরমান ম্যাকহোটেরিয়ান যে নীতিটি পরামর্শ দিচ্ছিল তার সাথে মেলে ।
এই সমাধানটি কাজ করার জন্য, আপনার লগ 4 নেট সংযোজনকারীদের আপনার কাস্টম ডেটা লগ করার জন্য আপনার কিছু অতিরিক্ত কনফিগারেশনও প্রয়োজন।
এই সমাধানটি কাস্টম লগ বর্ধন মডিউল হিসাবে সহজেই প্রয়োগ করা যেতে পারে। এখানে এর জন্য নমুনা কোডটি দেওয়া হল:
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();
}
}
}
এটি আপনার সাইটে যুক্ত করুন, আইআইএস 7+ কনফিডের নমুনা:
<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>