Correct way of using log4net (logger naming)

138,969

Solution 1

Regarding how you log messages within code, I would opt for the second approach:

ILog log = LogManager.GetLogger(typeof(Bar));
log.Info("message");

Where messages sent to the log above will be 'named' using the fully-qualifed type Bar, e.g.

MyNamespace.Foo.Bar [INFO] message

The advantage of this approach is that it is the de-facto standard for organising logging, it also allows you to filter your log messages by namespace. For example, you can specify that you want to log INFO level message, but raise the logging level for Bar specifically to DEBUG:

<log4net>
    <!-- appenders go here -->
    <root>
        <level value="INFO" />
        <appender-ref ref="myLogAppender" />
    </root>

    <logger name="MyNamespace.Foo.Bar">
        <level value="DEBUG" />
    </logger>
</log4net>

The ability to filter your logging via name is a powerful feature of log4net, if you simply log all your messages to "myLog", you loose much of this power!

Regarding the EPiServer CMS, you should be able to use the above approach to specify a different logging level for the CMS and your own code.

For further reading, here is a codeproject article I wrote on logging:

Solution 2

My Answer might be coming late, but I think it can help newbie. You shall not see logs executed unless the changes are made as below.

2 Files have to be changes when you implement Log4net.


  1. Add Reference of log4net.dll in the project.
  2. app.config
  3. Class file where you will implement Logs.

Inside [app.config] :

First, under 'configSections', you need to add below piece of code;

<section name="log4net" type="log4net.Config.Log4NetConfigurationSectionHandler, log4net" />

Then, under 'configuration' block, you need to write below piece of code.(This piece of code is customised as per my need , but it works like charm.)

<log4net debug="true">
    <logger name="log">
      <level value="All"></level>
      <appender-ref ref="RollingLogFileAppender" />
    </logger>

    <appender name="RollingLogFileAppender" type="log4net.Appender.RollingFileAppender">
      <file value="log.txt" />
      <appendToFile value="true" />
      <rollingStyle value="Composite" />
      <maxSizeRollBackups value="1" />
      <maximumFileSize value="1MB" />
      <staticLogFileName value="true" />

      <layout type="log4net.Layout.PatternLayout">
        <conversionPattern value="%date %C.%M [%line] %-5level - %message %newline %exception %newline" />
      </layout>
    </appender>
</log4net>

Inside Calling Class :

Inside the class where you are going to use this log4net, you need to declare below piece of code.

 ILog log = LogManager.GetLogger("log");

Now, you are ready call log wherever you want in that same class. Below is one of the method you can call while doing operations.

log.Error("message");

Solution 3

Instead of naming my invoking class, I started using the following:

private static readonly ILog log = LogManager.GetLogger(System.Reflection.MethodBase.GetCurrentMethod().DeclaringType);

In this way, I can use the same line of code in every class that uses log4net without having to remember to change code when I copy and paste. Alternatively, i could create a logging class, and have every other class inherit from my logging class.

Share:
138,969
Sly
Author by

Sly

Asp.net developer.

Updated on January 03, 2020

Comments

  • Sly
    Sly over 4 years

    There are two ways of configuring and using log4net. First one is when I can configure my own appender and associated logger:

    <!-- language: xml -->
    
    <appender name="myLogAppender" type="log4net.Appender.RollingFileAppender" >
        <file value="Logs\myLog.log" />
        <layout type="log4net.Layout.PatternLayout">
            <conversionPattern value="%date %level - %message%n" />
        </layout>
    </appender>
    
    <logger name="myLog">
        <level value="All"></level>
        <appender-ref ref="myLogAppender" />
    </logger>
    

    And then when I want to write something in log, I can do the following:

    ILog log = LogManager.GetLogger("myLog");
    log.Info("message");
    

    Another way to use it is to configure root to be as detailed as I want:

    <!-- language: xml -->
    
    <root>
        <level value="Error" />
        <appender-ref ref="myLogAppender" />
    </root>
    

    And in this case I can log messages like this:

    ILog log = LogManager.GetLogger(typeof(Bar));
    log.Info("message");
    

    The benefits of second approach is that you can enable or disable some messages on the fly. But the problem is that I'm developing in EPiServer CMS and it has its own logging system that uses log4net and if I enable info logging at root level, then a lot of system logs will be written.

    How do you use log4net? Each part of a system writes in its own logger, or everything is written in default logger, and configuration decides what to do next?

  • AlfeG
    AlfeG almost 13 years
    You can even exclude from log part of class namespaces to reduce noise in logs by using PatternLayout logging.apache.org/log4net/release/sdk/… "For example, for the logger name "a.b.c" the pattern %logger{2} will output "b.c"."
  • Casper Leon Nielsen
    Casper Leon Nielsen over 10 years
    private static readonly ILog log = LogManager.GetLogger(System.Reflection.MethodBase.GetCurrent‌​Method().DeclaringTy‌​pe);
  • MeTitus
    MeTitus over 8 years
    Why is the second approach better than the first?! The class name is static, and if you change it, you'll have to update the name in the logger too, is there any point in causing a reflection call just to get the class name?
  • ErikE
    ErikE over 7 years
    @CasperLeonNielsen Can you explain how that's different from this.GetType()?
  • dhochee
    dhochee over 7 years
    @ErikE this.GetType() won't be available when defining a static property, or in a static class, or outside a constructor.
  • ErikE
    ErikE over 7 years
    @dhochee quite right. I wasn't paying attention to the static nature of things. There is no this in a static method. What do you mean about "outside a constructor" though?
  • Minh Tran
    Minh Tran over 4 years
    You don't need to make ILog an instance-member right? I've asked the same question in more detail here but perhaps I can get your input?