Divergic.Logging.Xunit 4.3.1

Suggested Alternatives

Neovolve.Logging.Xunit

Additional Details

This package has been renamed to Neovolve.Logging.Xunit

dotnet add package Divergic.Logging.Xunit --version 4.3.1                
NuGet\Install-Package Divergic.Logging.Xunit -Version 4.3.1                
This command is intended to be used within the Package Manager Console in Visual Studio, as it uses the NuGet module's version of Install-Package.
<PackageReference Include="Divergic.Logging.Xunit" Version="4.3.1" />                
For projects that support PackageReference, copy this XML node into the project file to reference the package.
paket add Divergic.Logging.Xunit --version 4.3.1                
#r "nuget: Divergic.Logging.Xunit, 4.3.1"                
#r directive can be used in F# Interactive and Polyglot Notebooks. Copy this into the interactive tool or source code of the script to reference the package.
// Install Divergic.Logging.Xunit as a Cake Addin
#addin nuget:?package=Divergic.Logging.Xunit&version=4.3.1

// Install Divergic.Logging.Xunit as a Cake Tool
#tool nuget:?package=Divergic.Logging.Xunit&version=4.3.1                

Package rename

Note: This package is being renamed to Neovolve.Logging.Xunit.

Introduction

Divergic.Logging.Xunit is a NuGet package that returns an ILogger or ILogger<T> that wraps around the ITestOutputHelper supplied by xUnit. xUnit uses this helper to write log messages to the test output of each test execution. This means that any log messages from classes being tested will end up in the xUnit test result output.

GitHub license Nuget Nuget Actions Status

Installation

Run the following in the NuGet command line or visit the NuGet package page.

Install-Package Divergic.Logging.Xunit

Back to top

Usage

The common usage of this package is to call the BuildLogger<T> extension method on the xUnit ITestOutputHelper.

Consider the following example of a class to test.

using System;
using Microsoft.Extensions.Logging;

public class MyClass
{
    private readonly ILogger _logger;

    public MyClass(ILogger<MyClass> logger)
    {
        _logger = logger;
    }

    public string DoSomething()
    {
        _logger.LogInformation("Hey, we did something");

        return Guid.NewGuid().ToString();
    }
}

Call BuildLoggerFor<T>() on ITestOutputHelper to generate the ILogger<T> to inject into the class being tested.

public class MyClassTests
{
    private readonly ITestOutputHelper _output;

    public MyClassTests(ITestOutputHelper output)
    {
        _output = output;
    }

    [Fact]
    public void DoSomethingReturnsValue()
    {
        using var logger = output.BuildLoggerFor<MyClass>();

        var sut = new MyClass(logger);

        var actual = sut.DoSomething();

        // The xUnit test output should now include the log message from MyClass.DoSomething()

        actual.Should().NotBeNull();
    }
}

This would output the following in the test results.

Information [0]: Hey, we did something

Similarly, using the BuildLogger() extension method will return an ILogger configured with xUnit test output.

The above examples inline the declaration of the logger with using var to ensure that the logger instance (and internal ILoggerFactory) is disposed.

You can avoid having to build the logger instance in each unit test method by deriving the test class from either LoggingTestsBase or LoggingTestsBase<T>. These classes provide the implementation to build the logger and dispose it. They also provide access to the ITestOutputHelper instance for writing directly to the test output.

public class MyClassTests : LoggingTestsBase<MyClass>
{
    public MyClassTests(ITestOutputHelper output) : base(output, LogLevel.Information)
    {
    }

    [Fact]
    public void DoSomethingReturnsValue()
    {
        var sut = new MyClass(Logger);

        var actual = sut.DoSomething();

        // The xUnit test output should now include the log message from
        MyClass.DoSomething()

        Output.WriteLine("This works too");

        actual.Should().NotBeNullOrWhiteSpace();
    }
}

The BuildLogger and BuildLoggerFor<T> extension methods along with the LoggingTestsBase and LoggingTestsBase<T> abstract classes also provide overloads to set the logging level or define logging configuration.

Back to top

Output Formatting

The default formatting to the xUnit test results may not be what you want. You can define your own ILogFormatter class to control how the output looks. There is a configurable formatter for standard messages and another configurable formatter for scope start and end messages.

public class MyFormatter : ILogFormatter
{
    public string Format(
        int scopeLevel,
        string categoryName,
        LogLevel logLevel,
        EventId eventId,
        string message,
        Exception exception)
    {
        var builder = new StringBuilder();

        if (scopeLevel > 0)
        {
            builder.Append(' ', scopeLevel * 2);
        }

        builder.Append($"{logLevel} ");

        if (!string.IsNullOrEmpty(categoryName))
        {
            builder.Append($"{categoryName} ");
        }

        if (eventId.Id != 0)
        {
            builder.Append($"[{eventId.Id}]: ");
        }

        if (!string.IsNullOrEmpty(message))
        {
            builder.Append(message);
        }

        if (exception != null)
        {
            builder.Append($"\n{exception}");
        }

        return builder.ToString();
    }
}

public class MyConfig : LoggingConfig
{
    public MyConfig()
    {
        base.Formatter = new MyFormatter();
    }

    public static MyConfig Current { get; } = new MyConfig();
}

The custom ILogFormatter is defined on a LoggingConfig class that can be provided when creating a logger. The MyConfig.Current property above is there provide a clean way to share the config across test classes.

using System;
using FluentAssertions;
using Microsoft.Extensions.Logging;
using Xunit;
using Xunit.Abstractions;

public class MyClassTests
{
    private readonly ITestOutputHelper _output;

    public MyClassTests(ITestOutputHelper output)
    {
        _output = output;
    }

    [Fact]
    public void DoSomethingReturnsValue()
    {
        using var logger = _output.BuildLogger(MyConfig.Current);
        var sut = new MyClass(logger);

        var actual = sut.DoSomething();

        // The xUnit test output should now include the log message from MyClass.DoSomething()

        actual.Should().NotBeNull();
    }
}

In the same way the format of start and end scope messages can be formatted by providing a custom formatter on LoggingConfig.ScopeFormatter.

Back to top

Inspection

Using this library makes it really easy to output log messages from your code as part of the test results. You may want to also inspect the log messages written as part of the test assertions as well.

The BuildLogger and BuildLoggerFor<T> extension methods support this by returning a ICacheLogger or ICacheLogger<T> respectively. The cache logger is a wrapper around the created logger and exposes all the log entries written by the test.

using System;
using Divergic.Logging.Xunit;
using FluentAssertions;
using Microsoft.Extensions.Logging;
using Xunit;
using Xunit.Abstractions;

public class MyClassTests
{
    private readonly ITestOutputHelper _output;

    public MyClassTests(ITestOutputHelper output)
    {
        _output = output;
    }

    [Fact]
    public void DoSomethingReturnsValue()
    {
        using var logger = _output.BuildLogger();
        var sut = new MyClass(logger);

        sut.DoSomething();
        
        logger.Count.Should().Be(1);
        logger.Entries.Should().HaveCount(1);
        logger.Last.Message.Should().Be("Hey, we did something");
    }
}

Perhaps you don't want to use the xUnit ITestOutputHelper but still want to use the ICacheLogger to run assertions over log messages written by the class under test. You can do this by creating a CacheLogger or CacheLogger<T> directly.

using System;
using Divergic.Logging.Xunit;
using FluentAssertions;
using Microsoft.Extensions.Logging;
using Xunit;

public class MyClassTests
{
    [Fact]
    public void DoSomethingReturnsValue()
    {
        var logger = new CacheLogger();

        var sut = new MyClass(logger);

        sut.DoSomething();
        
        logger.Count.Should().Be(1);
        logger.Entries.Should().HaveCount(1);
        logger.Last.Message.Should().Be("Hey, we did something");
    }
}

Back to top

Configured LoggerFactory

You may have an integration or acceptance test that requires additional configuration to the log providers on ILoggerFactory while also supporting the logging out to xUnit test results. You can do this by create a factory that is already configured with xUnit support.

You can get an xUnit configured ILoggerFactory by calling output.BuildLoggerFactory().

using System;
using Divergic.Logging.Xunit;
using FluentAssertions;
using Microsoft.Extensions.Logging;
using Xunit;
using Xunit.Abstractions;

public class MyClassTests
{
    private readonly ILogger _logger;

    public MyClassTests(ITestOutputHelper output)
    {
        var factory = output.BuildLoggerFactory();

        // call factory.AddConsole or other provider extension method

        _logger = factory.CreateLogger(nameof(MyClassTests));
    }

    [Fact]
    public void DoSomethingReturnsValue()
    {
        var sut = new MyClass(_logger);

        // The xUnit test output should now include the log message from MyClass.DoSomething()

        var actual = sut.DoSomething();

        actual.Should().NotBeNullOrWhiteSpace();
    }
}

The BuildLoggerFactory extension methods provide overloads to set the logging level or define logging configuration.

Back to top

Existing Loggers

Already have an existing logger and want the above cache support? Got you covered there too using the WithCache() method.

using System;
using Divergic.Logging.Xunit;
using FluentAssertions;
using Microsoft.Extensions.Logging;
using NSubstitute;
using Xunit;

public class MyClassTests
{
    [Fact]
    public void DoSomethingReturnsValue()
    {
        var logger = Substitute.For<ILogger>();

        logger.IsEnabled(Arg.Any<LogLevel>()).Returns(true);

        var cacheLogger = logger.WithCache();

        var sut = new MyClass(cacheLogger);

        sut.DoSomething();

        cacheLogger.Count.Should().Be(1);
        cacheLogger.Entries.Should().HaveCount(1);
        cacheLogger.Last.Message.Should().Be("Hey, we did something");
    }
}

The WithCache() also supports ILogger<T>.

using System;
using Divergic.Logging.Xunit;
using FluentAssertions;
using Microsoft.Extensions.Logging;
using NSubstitute;
using Xunit;

public class MyClassTests
{
    [Fact]
    public void DoSomethingReturnsValue()
    {
        var logger = Substitute.For<ILogger<MyClass>>();

        logger.IsEnabled(Arg.Any<LogLevel>()).Returns(true);

        var cacheLogger = logger.WithCache();

        var sut = new MyClass(cacheLogger);

        sut.DoSomething();

        cacheLogger.Count.Should().Be(1);
        cacheLogger.Entries.Should().HaveCount(1);
        cacheLogger.Last.Message.Should().Be("Hey, we did something");
    }
}

Back to top

Sensitive Values

The LoggingConfig class exposes a SensitiveValues property that holds a collection of strings. All sensitive values found in a log message or a start/end scope message will be masked out.

public class ScopeScenarioTests : LoggingTestsBase<ScopeScenarioTests>
{
    private static readonly LoggingConfig _config = new LoggingConfig().Set(x => x.SensitiveValues.Add("secret"));

    public ScopeScenarioTests(ITestOutputHelper output) : base(output, _config)
    {
    }

    [Fact]
    public void TestOutputWritesScopeBoundariesUsingObjectsWithSecret()
    {
        Logger.LogCritical("Writing critical message with secret");
        Logger.LogDebug("Writing debug message with secret");
        Logger.LogError("Writing error message with secret");
        Logger.LogInformation("Writing information message with secret");
        Logger.LogTrace("Writing trace message with secret");
        Logger.LogWarning("Writing warning message with secret");

        var firstPerson = Model.Create<StructuredData>().Set(x => x.Email = "secret");

        using (Logger.BeginScope(firstPerson))
        {
            Logger.LogInformation("Inside first scope with secret");

            var secondPerson = Model.Create<StructuredData>().Set(x => x.FirstName = "secret");

            using (Logger.BeginScope(secondPerson))
            {
                Logger.LogInformation("Inside second scope with secret");
            }

            Logger.LogInformation("After second scope with secret");
        }

        Logger.LogInformation("After first scope with secret");
    }

The above test will render the following to the test output.

Critical [0]: Writing critical message with ****
Debug [0]: Writing debug message with ****
Error [0]: Writing error message with ****
Information [0]: Writing information message with ****
Trace [0]: Writing trace message with ****
Warning [0]: Writing warning message with ****
<Scope 1>
   Scope data: 
   {
     "DateOfBirth": "1972-10-07T16:35:31.2039449Z",
     "Email": "****",
     "FirstName": "Amos",
     "LastName": "Burton"
   }
   Information [0]: Inside first scope with ****
      <Scope 2>
      Scope data: 
      {
        "DateOfBirth": "1953-07-04T06:55:31.2333376Z",
        "Email": "james.holden@rocinante.space",
        "FirstName": "****",
        "LastName": "Holden"
      }
      Information [0]: Inside second scope with ****
   </Scope 2>
   Information [0]: After second scope with ****
</Scope 1>
Information [0]: After first scope with ****

Back to top

Configuration

Logging configuration can be controled by using a LoggingConfig class as indicated in the Output Formatting section above. The following are the configuration options that can be set.

Formatter: Defines a custom formatter for rendering log messages to xUnit test output.

ScopeFormatter: Defines a custom formatter for rendering start and end scope messages to xUnit test output.

IgnoreTestBoundaryException: Defines whether exceptions thrown while logging outside of the test execution will be ignored.

LogLevel: Defines the minimum log level that will be written to the test output. This helps to limit the noise in test output when set to higher levels. Defaults to LogLevel.Trace.

ScopePaddingSpaces: Defines the number of spaces to use for indenting scopes.

SensitiveValues: Defines a collection of sensitive values that will be masked in the test output logging.

Back to top

Product Compatible and additional computed target framework versions.
.NET net5.0 was computed.  net5.0-windows was computed.  net6.0 was computed.  net6.0-android was computed.  net6.0-ios was computed.  net6.0-maccatalyst was computed.  net6.0-macos was computed.  net6.0-tvos was computed.  net6.0-windows was computed.  net7.0 was computed.  net7.0-android was computed.  net7.0-ios was computed.  net7.0-maccatalyst was computed.  net7.0-macos was computed.  net7.0-tvos was computed.  net7.0-windows was computed.  net8.0 was computed.  net8.0-android was computed.  net8.0-browser was computed.  net8.0-ios was computed.  net8.0-maccatalyst was computed.  net8.0-macos was computed.  net8.0-tvos was computed.  net8.0-windows was computed.  net9.0 was computed.  net9.0-android was computed.  net9.0-browser was computed.  net9.0-ios was computed.  net9.0-maccatalyst was computed.  net9.0-macos was computed.  net9.0-tvos was computed.  net9.0-windows was computed. 
.NET Core netcoreapp2.0 was computed.  netcoreapp2.1 was computed.  netcoreapp2.2 was computed.  netcoreapp3.0 was computed.  netcoreapp3.1 was computed. 
.NET Standard netstandard2.0 is compatible.  netstandard2.1 was computed. 
.NET Framework net461 was computed.  net462 was computed.  net463 was computed.  net47 was computed.  net471 was computed.  net472 was computed.  net48 was computed.  net481 was computed. 
MonoAndroid monoandroid was computed. 
MonoMac monomac was computed. 
MonoTouch monotouch was computed. 
Tizen tizen40 was computed.  tizen60 was computed. 
Xamarin.iOS xamarinios was computed. 
Xamarin.Mac xamarinmac was computed. 
Xamarin.TVOS xamarintvos was computed. 
Xamarin.WatchOS xamarinwatchos was computed. 
Compatible target framework(s)
Included target framework(s) (in package)
Learn more about Target Frameworks and .NET Standard.

NuGet packages (6)

Showing the top 5 NuGet packages that depend on Divergic.Logging.Xunit:

Package Downloads
AutoMoqFixture

Package Description

U2U.EntityFrameworkCore.Testing

Package Description

InsonusK.Xunit.ExpectationsTest

Base class of Xunit test expectation methods support

Reductech.Sequence.Core.TestHarness

Class library for automatically testing Sequence® Core steps.

HerrGeneral.Test.Extension

Helper methods for HerrGeneral testing

GitHub repositories (9)

Showing the top 5 popular GitHub repositories that depend on Divergic.Logging.Xunit:

Repository Stars
bitwarden/server
Bitwarden infrastructure/backend (API, database, Docker, etc).
JasperFx/marten
.NET Transactional Document DB and Event Store on PostgreSQL
imperugo/StackExchange.Redis.Extensions
sethreno/schemazen
Script and create SQL Server objects quickly
josephwoodward/GlobalExceptionHandlerDotNet
Exception handling as a convention in the ASP.NET Core request pipeline
Version Downloads Last updated
4.3.1 289,076 4/20/2024 4.3.1 is deprecated.
4.3.0 534,612 11/21/2023 4.3.0 is deprecated.
4.2.0 1,893,980 8/9/2022 4.2.0 is deprecated.
4.2.0-beta0001 197 8/9/2022 4.2.0-beta0001 is deprecated.
4.1.0 163,025 6/23/2022 4.1.0 is deprecated.
4.1.0-beta0002 205 6/23/2022 4.1.0-beta0002 is deprecated.
4.1.0-beta0001 204 6/22/2022 4.1.0-beta0001 is deprecated.
4.0.0 555,754 1/10/2022 4.0.0 is deprecated.
3.6.1-beta0010 265 1/10/2022 3.6.1-beta0010 is deprecated.
3.6.0 1,240,153 11/13/2020 3.6.0 is deprecated.
3.5.2-beta0019 356 11/13/2020 3.5.2-beta0019 is deprecated.
3.5.2-beta0018 348 11/13/2020 3.5.2-beta0018 is deprecated.
3.5.2-beta0017 344 11/13/2020 3.5.2-beta0017 is deprecated.
3.5.2-beta0016 350 11/13/2020 3.5.2-beta0016 is deprecated.
3.5.2-beta0015 393 11/12/2020 3.5.2-beta0015 is deprecated.
3.5.2-beta0010 356 11/12/2020 3.5.2-beta0010 is deprecated.
3.5.2-beta0006 389 11/10/2020 3.5.2-beta0006 is deprecated.
3.5.2-beta0003 804 10/14/2020 3.5.2-beta0003 is deprecated.
3.5.2-beta0001 15,602 10/12/2020 3.5.2-beta0001 is deprecated.
3.5.1 136,593 10/8/2020 3.5.1 is deprecated.
3.5.1-releasejob0001 609 8/4/2020 3.5.1-releasejob0001 is deprecated.
3.5.1-beta0032 447 10/8/2020 3.5.1-beta0032 is deprecated.
3.5.1-beta0031 375 10/8/2020 3.5.1-beta0031 is deprecated.
3.5.1-beta0030 390 10/8/2020 3.5.1-beta0030 is deprecated.
3.5.1-beta0029 461 9/27/2020 3.5.1-beta0029 is deprecated.
3.5.1-beta0028 395 9/15/2020 3.5.1-beta0028 is deprecated.
3.5.1-beta0027 400 9/9/2020 3.5.1-beta0027 is deprecated.
3.5.1-beta0026 394 9/9/2020 3.5.1-beta0026 is deprecated.
3.5.1-beta0025 395 9/2/2020 3.5.1-beta0025 is deprecated.
3.5.1-beta0024 384 8/21/2020 3.5.1-beta0024 is deprecated.
3.5.1-beta0023 431 8/15/2020 3.5.1-beta0023 is deprecated.
3.5.1-beta0022 437 8/12/2020 3.5.1-beta0022 is deprecated.
3.5.1-beta0021 378 8/12/2020 3.5.1-beta0021 is deprecated.
3.5.1-beta0020 406 8/11/2020 3.5.1-beta0020 is deprecated.
3.5.1-beta0019 432 8/8/2020 3.5.1-beta0019 is deprecated.
3.5.1-beta0018 439 8/8/2020 3.5.1-beta0018 is deprecated.
3.5.1-beta0017 447 8/8/2020 3.5.1-beta0017 is deprecated.
3.5.1-beta0015 417 8/5/2020 3.5.1-beta0015 is deprecated.
3.5.1-beta0013 391 8/4/2020 3.5.1-beta0013 is deprecated.
3.5.1-beta0012 397 8/4/2020 3.5.1-beta0012 is deprecated.
3.5.1-beta0011 409 8/4/2020 3.5.1-beta0011 is deprecated.
3.5.1-beta0010 414 8/2/2020 3.5.1-beta0010 is deprecated.
3.5.1-beta0009 442 7/31/2020 3.5.1-beta0009 is deprecated.
3.5.1-beta0008 446 7/31/2020 3.5.1-beta0008 is deprecated.
3.5.1-beta0007 450 7/31/2020 3.5.1-beta0007 is deprecated.
3.5.1-beta0003 454 7/31/2020 3.5.1-beta0003 is deprecated.
3.5.0 188,757 6/22/2020 3.5.0 is deprecated.
3.5.0-beta0001 420 6/17/2020 3.5.0-beta0001 is deprecated.
3.4.0 123,484 4/28/2020 3.4.0 is deprecated.
3.3.1-beta0004 417 4/28/2020 3.3.1-beta0004 is deprecated.
3.3.0 61,784 3/18/2020 3.3.0 is deprecated.
3.3.0-beta0005 513 3/18/2020 3.3.0-beta0005 is deprecated.
3.2.2-beta0003 2,817 1/12/2020 3.2.2-beta0003 is deprecated.
3.2.2-beta0002 479 1/12/2020 3.2.2-beta0002 is deprecated.
3.2.2-beta0001 476 1/12/2020 3.2.2-beta0001 is deprecated.
3.2.1 226,850 12/18/2019 3.2.1 is deprecated.
3.2.1-beta0001 504 12/18/2019 3.2.1-beta0001 is deprecated.
3.2.0 718 12/18/2019 3.2.0 is deprecated.
3.2.0-beta0002 497 12/18/2019 3.2.0-beta0002 is deprecated.
3.1.0 201,572 9/27/2019 3.1.0 is deprecated.
3.1.0-beta0002 469 9/27/2019 3.1.0-beta0002 is deprecated.
3.0.0 74,496 5/3/2019 3.0.0 is deprecated.
2.2.0 54,548 4/18/2019 2.2.0 is deprecated.
2.2.0-beta0001 532 4/18/2019 2.2.0-beta0001 is deprecated.
2.1.0 1,837 4/14/2019 2.1.0 is deprecated.
2.1.0-beta0003 551 4/14/2019 2.1.0-beta0003 is deprecated.
2.1.0-beta0002 517 4/13/2019 2.1.0-beta0002 is deprecated.
2.0.1-beta0001 567 3/25/2019 2.0.1-beta0001 is deprecated.
2.0.0 47,150 3/19/2019 2.0.0 is deprecated.
1.1.0 53,849 6/3/2018 1.1.0 is deprecated.
1.0.0 9,575 6/1/2018 1.0.0 is deprecated.
0.2.0-beta0047 401 8/5/2020 0.2.0-beta0047 is deprecated.
0.2.0-beta0045 389 8/4/2020 0.2.0-beta0045 is deprecated.
0.2.0-beta0036 400 7/31/2020 0.2.0-beta0036 is deprecated.
0.2.0-beta0035 405 7/31/2020 0.2.0-beta0035 is deprecated.
0.2.0-beta0034 455 6/22/2020 0.2.0-beta0034 is deprecated.