Implementing Logging in C# 9.0

Introduction

In software development, logging is an essential practice for monitoring and troubleshooting applications. It provides valuable insights into the application's behavior and helps identify and diagnose issues. With C# 9.0 and .NET 5, logging has become even more robust, offering developers a range of tools and features to implement effective logging mechanisms. This article delves into logging in C# 9.0, highlighting its importance, exploring various logging frameworks, and providing practical code examples to demonstrate its implementation.

The Importance of Logging

Logging serves multiple purposes in software development. It helps in:

  • Debugging: By capturing detailed information about the application's execution, logging aids in pinpointing the root cause of issues.
  • Monitoring: Logs provide real-time data about the application's performance and health, enabling proactive monitoring.
  • Auditing: Logging records significant events and actions, which can be crucial for auditing and compliance purposes.
  • Maintenance: Logs offer historical data, making it easier to understand the application's past behavior and make informed decisions during maintenance.

Logging Frameworks in C# 9.0

C# 9.0 and .NET 8.0 support a variety of logging frameworks, each offering unique features. Some of the popular logging frameworks include:

  • Microsoft.Extensions.Logging: This is the built-in logging framework in .NET, offering a flexible and extensible logging infrastructure.
  • NLog: A popular logging framework is known for its rich features, ease of use, and extensive configuration options.
  • Serilog: A structured logging framework that enables developers to write logs in a structured format, making it easier to query and analyze.

Setting Up Logging in C# 9.0

To get started with logging in a C# 9.0 application, you need to set up a logging framework. Here, we will use the built-in Microsoft.Extensions.Logging framework.

Step 1. Create a new .NET 9.0 console application

dotnet new console -n LoggingDemo
cd LoggingDemo

Step 2. Add the required logging packages:

dotnet add package Microsoft.Extensions.Logging
dotnet add package Microsoft.Extensions.Logging.Console

Step 3. Configuring Logging

Configure logging in your application by setting up the logging provider and creating a logger. Update the Program.cs file as follows:

using System;
using Microsoft.Extensions.Logging;

class Program
{
    static void Main(string[] args)
    {
        using var loggerFactory = LoggerFactory.Create(builder =>
        {
            builder
                .AddConsole()
                .SetMinimumLevel(LogLevel.Information);
        });

        ILogger logger = loggerFactory.CreateLogger<Program>();

        logger.LogInformation("Application started.");

        try
        {
            PerformTask(logger);
        }
        catch (Exception ex)
        {
            logger.LogError(ex, "An error occurred while performing the task.");
        }

        logger.LogInformation("Application ended.");
    }

    static void PerformTask(ILogger logger)
    {
        logger.LogDebug("Performing task...");

        // Simulate a task
        for (int i = 0; i < 5; i++)
        {
            logger.LogInformation($"Task iteration {i + 1}");
            System.Threading.Thread.Sleep(1000);
        }

        logger.LogDebug("Task completed successfully.");
    }
}

Using NLog for Advanced Logging : 

Step 1. First, add the NLog packages to your project:

dotnet add package NLog
dotnet add package NLog.Extensions.Logging

Step 2. Configure NLog by creating a nlog.config file with the following content:

<nlog xmlns="http://www.nlog-project.org/schemas/NLog.xsd"
      xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">
  <targets>
    <target xsi:type="File" name="logfile" fileName="logfile.txt" />
    <target xsi:type="Console" name="logconsole" />
  </targets>
  <rules>
    <logger name="*" minlevel="Info" writeTo="logfile,logconsole" />
  </rules>
</nlog>

Step 3. Update the Program.cs file to use NLog.

using System;
using NLog;
using NLog.Extensions.Logging;

class Program
{
    private static readonly Logger logger = LogManager.GetCurrentClassLogger();

    static void Main(string[] args)
    {
        var config = new NLogLoggingConfiguration(new NLog.Config.XmlLoggingConfiguration("nlog.config"));
        LogManager.Configuration = config;

        logger.Info("Application started.");

        try
        {
            PerformTask();
        }
        catch (Exception ex)
        {
            logger.Error(ex, "An error occurred while performing the task.");
        }

        logger.Info("Application ended.");
        LogManager.Shutdown();
    }

    static void PerformTask()
    {
        logger.Debug("Performing task...");

        // Simulate a task
        for (int i = 0; i < 5; i++)
        {
            logger.Info($"Task iteration {i + 1}");
            System.Threading.Thread.Sleep(1000);
        }

        logger.Debug("Task completed successfully.");
    }
}

Structured Logging with Serilog :

Structured logging is a powerful feature that allows developers to write logs in a structured format, making them easier to query and analyze. Serilog is a popular framework for structured logging.

Step 1. To get started with Serilog, add the necessary packages:

dotnet add package Serilog
dotnet add package Serilog.Extensions.Logging
dotnet add package Serilog.Sinks.Console

Step 2. Update the Program.cs file to use Serilog.

using System;
using Serilog;

class Program
{
    static void Main(string[] args)
    {
        Log.Logger = new LoggerConfiguration()
            .WriteTo.Console()
            .CreateLogger();

        Log.Information("Application started.");

        try
        {
            PerformTask();
        }
        catch (Exception ex)
        {
            Log.Error(ex, "An error occurred while performing the task.");
        }

        Log.Information("Application ended.");
        Log.CloseAndFlush();
    }

    static void PerformTask()
    {
        Log.Debug("Performing task...");

        // Simulate a task
        for (int i = 0; i < 5; i++)
        {
            Log.Information("Task iteration {Iteration}", i + 1);
            System.Threading.Thread.Sleep(1000);
        }

        Log.Debug("Task completed successfully.");
    }
}
using System;
using Serilog;

class Program
{
    static void Main(string[] args)
    {
        Log.Logger = new LoggerConfiguration()
            .WriteTo.Console()
            .CreateLogger();

        Log.Information("Application started.");

        try
        {
            PerformTask();
        }
        catch (Exception ex)
        {
            Log.Error(ex, "An error occurred while performing the task.");
        }

        Log.Information("Application ended.");
        Log.CloseAndFlush();
    }

    static void PerformTask()
    {
        Log.Debug("Performing task...");

        // Simulate a task
        for (int i = 0; i < 5; i++)
        {
            Log.Information("Task iteration {Iteration}", i + 1);
            System.Threading.Thread.Sleep(1000);
        }

        Log.Debug("Task completed successfully.");
    }
}
using System;
using Serilog;

class Program
{
    static void Main(string[] args)
    {
        Log.Logger = new LoggerConfiguration()
            .WriteTo.Console()
            .CreateLogger();

        Log.Information("Application started.");

        try
        {
            PerformTask();
        }
        catch (Exception ex)
        {
            Log.Error(ex, "An error occurred while performing the task.");
        }

        Log.Information("Application ended.");
        Log.CloseAndFlush();
    }

    static void PerformTask()
    {
        Log.Debug("Performing task...");

        // Simulate a task
        for (int i = 0; i < 5; i++)
        {
            Log.Information("Task iteration {Iteration}", i + 1);
            System.Threading.Thread.Sleep(1000);
        }

        Log.Debug("Task completed successfully.");
    }
}

Conclusion

Logging is an indispensable part of software development, providing insights into application behavior, aiding in debugging, and ensuring effective monitoring. With C# 9.0 and .NET 8.0, developers have access to a variety of powerful logging frameworks, including Microsoft.Extensions.Logging, NLog, and Serilog. Each of these frameworks offers unique features and capabilities, enabling developers to implement robust and efficient logging mechanisms tailored to their specific needs. By incorporating logging into your applications, you can significantly enhance their reliability, maintainability, and overall performance.


Similar Articles