Logging .NET apps using NLog
Debugging messages, logs, can become very messy very quickly in development teams. I've decided to set up a sort of logging framework in my last team. Here is what I did.
Nlog
I imported the nuget package NLog : https://nlog-project.org/
"NLog is a flexible and free logging platform for various .NET platforms, including .NET standard. NLog makes it easy to write to several targets. (database, file, console) and change the logging configuration on-the-fly."
- https://nlog-project.org/
See the documentation for more information : https://github.com/nlog/nlog/wiki
Supported platforms:
- .NET Framework 3.5, 4, 4.5, 4.6 & 4.7
- .NET Standard 1.3+ and 2.0+;
- .NET Framework 4 client profile
- Xamarin.Android, Xamarin.iOS, Xamarin.Forms
- UWP
- Windows Phone 8
- Silverlight 4 and 5
- Mono 4
For ASP.NET Core, check: https://www.nuget.org/packages/NLog.Web.AspNetCore
Initialisation
First we need to initialise NLog. Here is my initialisation method :
private static Logger _logger; //NLog logger
public static void Initialize()
{
_logger = LogManager.GetLogger("Default") ?? LogManager.GetCurrentClassLogger();
var config = new NLog.Config.LoggingConfiguration();
var logFile = new NLog.Targets.FileTarget()
{
FileName = "nlog.log",
Name = "logfile",
Layout = "${level:upperCase=true}:${message}${exception:format=ToString}"
};
var logConsole = new NLog.Targets.ColoredConsoleTarget()
{
Name = "logconsole",
Layout = "${level:upperCase=true}:${message}"
};
config.LoggingRules.Add(new NLog.Config.LoggingRule("*", LogLevel.Trace, logConsole));
config.LoggingRules.Add(new NLog.Config.LoggingRule("*", LogLevel.Debug, logFile));
LogManager.Configuration = config;
typeof(Logging).Trace($"Logging started. Log file: {logFile.FileName}");
}
You need to call it as soon as your app is ready, in the Program.Main(); AppDelegate.FinishedLaunching(); for Xamarin.iOS or MainActivity.OnCreate(); for Xamarin.Android.
LoggingHelper.Initialize();
Extension methods
You might ask why the Initialisation method is a static method. Simply because I've put it in a static class in order to define extension methods.
Pros
- Accessible from every class of the project without having to create huge inheritance tree (core object that all objects would need to inherit from)
- Passes class, method and line of the caller to the logging system automatically through System.Runtime.CompilerServices
Cons
- Does not exist on a static class or in a static method. It needs a this. See the workaround below.
BuildLogLine function
The function BuildLogLine is where the concatenation happens. Feel free to tweak it. You can add the app's uptime, the file line etc.
private static string BuildLogLine<T>(T caller, string message = "", string callerFunction = "")
{
StringBuilder sb = new StringBuilder();
var type = caller as Type;
var callerName = (type != null ? type : caller.GetType()).Name;
sb.Append($@" {callerName}");
if (!String.IsNullOrWhiteSpace(callerFunction))
sb.Append($@".{callerFunction}");
if (!String.IsNullOrWhiteSpace(message))
sb.Append($@": {message}");
return sb.ToString();
}
Code here : Logging.cs
Usage
Regular usage
// this.Info, this.Trace and this.Warn can be used at any point in the code.
this.Info("For your information, this is happening right now");
this.Trace("Here is some raw data I received at this point : { 'name': 'yolo' }");
this.Warn("I don't like this, something weird happened ...");
Static classes and functions
public static class MyStaticClass
{
public static void MyStaticFunction()
{
// Since the keyword "this" is not available in a static context we need to use a ruse.
// Types are objects, a static class may not have an instance but it has a type !
typeof(MyStaticClass).Info("For your information, this is happening right now");
typeof(MyStaticClass).Trace("Here is some raw data I received at this point : { 'name': 'yolo' }");
typeof(MyStaticClass).Warn("I don't like this, something weird happened ...");
}
}
Errors and Exceptions
try
{
myObject.MyMethod(); // throws an exception
}
catch (Exception ex)
{
// this.Warn can take an exception parameter too.
// This is useful for logging predictable/expected exceptions.
typeof(Program).Warn("I knew something had happened !", ex);
// this.Error always needs an exception.
// This forces all developers to handle errors with exceptions.
typeof(Program).Error("Something definitely broke there", ex);
}
Output
A file "nlog.log" has been created with only Info, Warn and Error. This is on purpose so if you save this log on a mobile device and send it (crash report) afterward it's not polluted with Traces.
INFO: MyClass.MyMethod: For your information, this is happening right now
WARN: MyClass.MyMethod: I don't like this, something weird happened ...
WARN: Program.Main: I knew something had happened !
System.Exception: Nope
at LoggingDemo.MyClass.ThrowException() in /Users/framinosona/Projects/LoggingDemo/MyClass.cs:line 22
at LoggingDemo.Program.Main(String[] args) in /Users/framinosona/Projects/LoggingDemo/Program.cs:line 20
ERROR: Program.Main: Something definitely broke there
System.Exception: Nope
at LoggingDemo.MyClass.ThrowException() in /Users/framinosona/Projects/LoggingDemo/MyClass.cs:line 22
at LoggingDemo.Program.Main(String[] args) in /Users/framinosona/Projects/LoggingDemo/Program.cs:line 20
INFO: MyStaticClass.MyStaticFunction: For your information, this is happening right now
WARN: MyStaticClass.MyStaticFunction: I don't like this, something weird happened ...
And the terminal is more detailed, for debugging purposes :
TRACE: LoggingHelper.Initialize: Logging started. Log file: 'nlog.log'
INFO: MyClass.MyMethod: For your information, this is happening right now
TRACE: MyClass.MyMethod: Here is some raw data I received at this point : { 'name': 'yolo' }
WARN: MyClass.MyMethod: I don't like this, something weird happened ...
TRACE: MyClass.MyMethod
WARN: Program.Main: I knew something had happened !
ERROR: Program.Main: Something definitely broke there
INFO: MyStaticClass.MyStaticFunction: For your information, this is happening right now
TRACE: MyStaticClass.MyStaticFunction: Here is some raw data I received at this point : { 'name': 'yolo' }
WARN: MyStaticClass.MyStaticFunction: I don't like this, something weird happened ...
Here is the full Gist : https://gist.github.com/framinosona/a0eadd93888f84f50a925b1e79b588a8
Tell me if this was helpful ! 👍