NLog Configuration Options
NLog can be configured in many ways, both programmatically and using config files. This document describes the available configuration options.
Logging configuration
NLog attempts to automatically configure itself on startup, by looking for the configuration files in some standard places. The following locations will be searched when executing a stand-alone *.exe application:
- standard application configuration file (usually applicationname.exe.config)
- applicationname.exe.nlog in application's directory
- NLog.config in application's directory
- NLog.dll.nlog in a directory where NLog.dll is located
- file name pointed by the NLOG_GLOBAL_CONFIG_FILE environment variable (if defined)
In case of an ASP.NET application, the following files are searched:
- standard web application file web.config
- web.nlog located in the same directory as web.config
- NLog.config in application's directory
- NLog.dll.nlog in a directory where NLog.dll is located
- file name pointed by the NLOG_GLOBAL_CONFIG_FILE environment variable (if defined)
The .NET Compact Framework doesn't recognize application configuration files (*.exe.config) nor environmental variables, so NLog only looks in these locations:
- applicationname.exe.nlog in application's directory
- NLog.config in application's directory
- NLog.dll.nlog in a directory where NLog.dll is located
Configuration file format
NLog supports two configuration file formats:
- configuration embedded within the standard *.exe.config or web.config file
- simplified configuration, stored in a separate file
In the first variant, we use a standard configSections mechanism, which makes our file look like this:
<configuration>
<configSections>
<section name="nlog" type="NLog.Config.ConfigSectionHandler, NLog"/>
</configSections>
<nlog>
</nlog>
</configuration>
The simplified format is the pure XML having the <nlog /> element as its root. The use of namespaces is optional, but it enables the Intellisense in Visual Studio.
<nlog xmlns="http://www.nlog-project.org/schemas/NLog.xsd"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">
</nlog>
Note that NLog config file is case-insensitive when not using namespaces and is case-sensitive when you use them. Intellisense only works with case-sensitive configurations.
Configuration elements
You can use the following elements as children to <nlog />. The first two elements from the list are required to be present in all NLog configuration files, the remaining ones are optional and can be useful in advanced scenarios.
- <targets /> - defines log targets/outputs
- <rules /> - defines log routing rules
- <extensions /> - loads NLog extensions from the *.dll file
- <include /> - includes external configuration file
- <variable /> - sets the value of a configuration variable
Targets
The <targets /> section defines log targets/outputs. Each target is represented by a <target /> element. There are two attributes required for each target:
- name - target name
- type - target type - such as "File", "Database", "Mail". When using namespaces this attribute is named xsi:type.
In addition to these attributes, targets usually accept other parameters, which influence the way diagnostic traces are written. Each target has a different set of parameters, they are described in detail on project's homepage and they are context-sensitive. Intellisense is also available in Visual Studio.
For example - the "File" target accepts the fileName parameter which defines output file name and the Console target has the error parameter which determines whether the diagnostic traces are written to standard error (stderr) instead of standard output (stdout) of the process.
This example demonstrates a <targets /> section which defines multiple targets: two files, one network target and OutputDebugString target:
<targets>
<target name="f1" xsi:type="File" fileName="file1.txt"/>
<target name="f2" xsi:type="File" fileName="file2.txt"/>
<target name="n1" xsi:type="Network" address="tcp://localhost:4001"/>
<target name="ds" xsi:type="OutputDebugString"/>
</targets>
NLog provides many predefined targets. They are described on the project's homepage. It's actually very easy to create your own target - it requires about 15-20 lines of code and is described in the documentation.
Rules
Log routing rules are defined in the <rules /> section. It is a simple routing table, where we define the list of targets that should be written to for each combination of source/logger name and log level. Rules are processed starting with the first rule in the list. When a rule matches, log messages are directed to target(s) in that rule. If a rule is marked as final, rules beneath the current rule are not processed.
Each routing table entry is a <logger /> element, which accepts the following attributes:
- name - source/logger name (may include wildcard characters *)
- minlevel - minimal log level for this rule to match
- maxlevel - maximum log level for this rule to match
- level - single log level for this rule to match
- levels - comma separated list of log levels for this rule to match
- writeTo - comma separated list of target that should be written to when this rule matches.
- final - make this rule final. No further rules are processed when any final rule matches.
Some examples:
- <logger name="Name.Space.Class1" minlevel="Debug" writeTo="f1" /> - all messages from the Class1 in the Name.Space whose level is Debug or higher are written to the "f1" target.
- <logger name="Name.Space.Class1" levels="Debug,Error" writeTo="f1" /> - all messages from the Class1 in the Name.Space whose level is either Debug or Error or higher are written to the "f1" target.
- <logger name="Name.Space.*" writeTo="f3,f4" /> - messages from any class in the Name.Space namespace are written to both "f3" and "f4" targets regardless of their levels.
- <logger name="Name.Space.*" minlevel="Debug" maxlevel="Error" final="true" /> - messages from any class in the Name.Space namespace whose level is between Debug and Error (which makes it Debug,Info,Warn,Error) are rejected (as there's no writeTo clause) and no futher rules are processed for them (because of the final="true" setting)
In the simplest cases the entire logging configuration consists of a single <target /> and a single <logger /> rule that routes messages to this target depending on their level. As the application grows, adding more targets and rules is very simple.
Contextual information
One of NLog's strongest assets is the ability to use layouts. They include pieces of text surrounded by a pair of "${" (dollar sign + left curly brace) and "}" (right curly brace). The markup denotes "layout renderers" which can be used to insert pieces of contextual information into the text. Layouts can be used in many places, for example they can control the format of information written on the screen or sent to a file, but also to control the file names themselves. This is very powerful, which we'll see in a moment.
Let's assume, that we want to augment each message written to the console with:
- current date and time
- name of the class and method that emitted the log message
- log level
- message text
This is very easy:
<target name="c" xsi:type="Console"
layout="${longdate} ${callsite} ${level} ${message}"/>
We can make each messages for each logger go to a separate file, as in the following example:
<target name="f" xsi:type="File" fileName="${logger}.txt"/>
Ay you can see the ${logger} layout renderer was used in the fileName attribute, which causes each log message to be written to the file whose name includes the logger name. The above example will create the following files:
- Name.Space.Class1.txt
- Name.Space.Class2.txt
- Name.Space.Class3.txt
- Other.Name.Space.Class1.txt
- Other.Name.Space.Class2.txt
- Other.Name.Space.Class3.txt
- ...
It's a frequent requirement to be able to keep log files for each day separate. This is trivial, too, thanks to the ${shortdate} layout renderer:
<target name="f" xsi:type="File" fileName="${shortdate}.txt"/>
How about givin each employee their own log file? The ${windows-identity} layout renderer will do the trick:
<target name="f" xsi:type="File" fileName="${windows-identity:domain=false}.txt"/>
Thanks to this simple setting NLog will create a set of files named after our employees' logins:
- Administrator.txt
- MaryManager.txt
- EdwardEmployee.txt
- ...
More complex cases are of course possible. The following sample demonstrates the way of creating a distinct log file for each person per day. Log files for each day are stored in a separate directory:
<target name="f" xsi:type="File"
fileName="${shortdate}/${windows-identity:domain=false}.txt"/>
This creates the following files:
- 2006-01-01/Administrator.txt
- 2006-01-01/MaryManager.txt
- 2006-01-01/EdwardEmployee.txt
- 2006-01-02/Administrator.txt
- 2006-01-02/MaryManager.txt
- 2006-01-02/EdwardEmployee.txt
- ...
NLog provides many predefined layout renderers. They are described on the http://www.nlog-project.org/layoutrenderers.html page. It's very easy to create your own layout renderer. It just takes 15-20 lines of code and is described in the documentation section of the project website.
Include files
It's sometimes desired to split the configuration file into many smaller ones. NLog provides an include file mechanism for that. To include an external file, you simply use It's worth noting that the fileName attribute, just like most attributes in NLog config file(s), may include dynamic values using the familiar ${var} notation, so it's possible to include different files based on environmental properties. The following configuration example demonstrates this, by loading file whose name is derived from the name of the machine we're running on.
<nlog>
...
<include file="${basedir}/${machinename}.config"/>
...
</nlog>
Variables let us write complex or repeatable expression (such as file names) in a concise manner. To define a variable we use the <variable name="var" value="xxx" /> syntax. Once defined, variables can be used as if they were layout renderers - by using ${var} syntax, as demonstrated in the following example:
<nlog>
<variable name="logDirectory" value="${basedir}/logs/${shortdate}"/>
<targets>
<target name="file1" xsi:type="File" fileName="${logDirectory}/file1.txt"/>
<target name="file2" xsi:type="File" fileName="${logDirectory}/file2.txt"/>
</targets>
</nlog>
Automatic reconfiguration
The configuration file is read automatically at program startup. In a long running process (such as a Windows service or an ASP.NET application) it's sometimes desirable to temporarily increase the log level without stopping the application. NLog can monitor logging configuration files and re-read them each time they are modified. To enable this mechanism, you simply set <nlog autoReload="true" /> in your configuration file. Note that automatic reconfiguration supports include files, so each time one of the files include is changed, the entire configuration gets reloaded.
Troubleshooting logging
Sometimes our application doesn't write anything to the log files, even though we have supposedly configured logging properly. There can be many reasons for logs not being written. The most common problem are permissions, usually in an ASP.NET process, where the aspnet_wp.exe or w3wp.exe process may not have write access to the directory where we want to store logs. NLog is designed to swallow run-time exceptions that may result from logging. The following settings can change this behavior and/or redirect these messages.
- <nlog throwExceptions="true" />- adding throwExceptions attribute in the config file causes NLog not to mask exceptions and pass them to the calling application instead. This attribute is useful at deployment time to quickly locate any problems. It's recommended to turn throwExceptions to "false" as soon as the application is properly configured to run, so that any accidental logging problems won't crash the application.
- <nlog internalLogFile="file.txt" />- adding internalLogFile cause NLog to write its internal debugging messages to the specified file. This includes any exceptions that may be thrown during logging.
- <nlog internalLogLevel="Trace|Debug|Info|Warn|Error|Fatal" /> - determines internal log level. The higher the level, the less verbose the internal log output.
- <nlog internalLogToConsole="false|true" /> - sends internal logging messages to the console.
- <nlog internalLogToConsoleError="false|true" /> - sends internal logging messages to the console error output (stderr).
Asynchronous processing, wrapper and compound targets
NLog provides wrapper and compound targets which modify other targets' behaviour by adding features like:
- asynchronous processing (wrapped target runs in a separate thread)
- retry-on-error
- load balancing (round-robin targets)
- buffering
- filtering
- backup targets (failover)
- and others described on http://www.nlog-project.org/targets.html
To define a wrapper or compound target in the configuration file, simply nest a target node within another target node. You can even wrap a wrapper target. There are no limits on depth. For example, to add asynchronous logging with retry-on-error functionality add this to your configuration file:
<targets>
<target name="n" xsi:type="AsyncWrapper">
<target xsi:type="RetryingWrapper">
<target xsi:type="File" fileName="${file}.txt"/>
</target>
</target>
</targets>
Because asynchronous processing is a common scenario, NLog supports a shorthand notation to enable it for all targets without the need to specify explicit wrappers. You simply set <targets async="true" /> and all your targets will be wrapped with the AsyncWrapper target.
Default wrappers
Sometimes we require ALL targets to be wrapped in the same way, for example to add buffering and/or retrying. NLog provides <default-wrapper /> syntax for that. You simply put this element in the <targets /> section and all your targets will be automatically wrapped with the specified wrapper. Note that <default-wrapper /> applies to the single <targets /> section only and you can have multiple sections so you can define groups of targets that are wrapped in a similar manner.
<nlog>
<targets>
<default-wrapper xsi:type="BufferingWrapper" bufferSize="100"/>
<target name="f1" xsi:type="File" fileName="f1.txt"/>
<target name="f2" xsi:type="File" fileName="f2.txt"/>
</targets>
<targets>
<default-wrapper xsi:type="AsyncWrapper">
<wrapper xsi:type="RetryingWrapper"/>
</default-wrapper>
<target name="n1" xsi:type="Network" address="tcp://localhost:4001"/>
<target name="n2" xsi:type="Network" address="tcp://localhost:4002"/>
<target name="n3" xsi:type="Network" address="tcp://localhost:4003"/>
</targets>
</nlog>
In the above example we've defined two buffered File targets and three asynchronous and retrying Network targets.
Default target parameters
Similar to default wrappers NLog provides <default-target-parameters /> which enables you to specify default values of target parameters. For example, if you don't want files to be kept open, you can either add keepFileOpen="false" to each target, as in the following example:
<nlog>
<targets>
<target name="f1" xsi:type="File" fileName="f1.txt" keepFileOpen="false"/>
<target name="f2" xsi:type="File" fileName="f2.txt" keepFileOpen="false"/>
<target name="f3" xsi:type="File" fileName="f3.txt" keepFileOpen="false"/>
</targets>
</nlog>
Alternatively you can specify a single <default-target-parameters /> that applies to all targets in the <targets /> section. Default parameters are defined on a per-type basis and are applied BEFORE the actual attributes defined in the XML file:
<nlog>
<targets>
<default-target-parameters xsi:type="File" keepFileOpen="false"/>
<target name="f1" xsi:type="File" fileName="f1.txt"/>
<target name="f2" xsi:type="File" fileName="f2.txt"/>
<target name="f3" xsi:type="File" fileName="f3.txt"/>
</targets>
</nlog>


