Contrib.Xunit.ActivityListenerTestFramework
2.0.0
dotnet add package Contrib.Xunit.ActivityListenerTestFramework --version 2.0.0
NuGet\Install-Package Contrib.Xunit.ActivityListenerTestFramework -Version 2.0.0
<PackageReference Include="Contrib.Xunit.ActivityListenerTestFramework" Version="2.0.0" />
paket add Contrib.Xunit.ActivityListenerTestFramework --version 2.0.0
#r "nuget: Contrib.Xunit.ActivityListenerTestFramework, 2.0.0"
// Install Contrib.Xunit.ActivityListenerTestFramework as a Cake Addin #addin nuget:?package=Contrib.Xunit.ActivityListenerTestFramework&version=2.0.0 // Install Contrib.Xunit.ActivityListenerTestFramework as a Cake Tool #tool nuget:?package=Contrib.Xunit.ActivityListenerTestFramework&version=2.0.0
Contrib.Xunit.ActivityListenerTestFramework
This allows you to test the branching paths of tracing using System.Diagnostics.Activity
Motivation
When creating observable libraries or applications you will very likely be using intrumenting your code using System.Diagnostics.Activity
and System.Diagnostics.ActivitySource
, this creates a situation that when there are no listeners your activity source will create null activities and the code will have many null check branches like the following
//Sample taken from: https://learn.microsoft.com/en-us/dotnet/core/diagnostics/observability-otlp-example#3-create-an-api-endpoint
async Task<String> SendGreeting(ILogger<Program> logger)
{
// Create a new Activity scoped to the method
using var activity = greeterActivitySource.StartActivity("GreeterActivity");
// Add a tag to the Activity
activity?.SetTag("greeting", "Hello World!"); //Null conditional method call
return "Hello World!";
}
The nullability check is a branch in the code execution and will be marked as a partially covered line in most code coverage reports unless the method is tested once while listening to the activity and once while not.
Unfortunately the activiy source and activity listeners are considered static scope thus creating a challenge when you want to achieve higher code coverage and these branching paths could actually be a source of bugs if not tested correctly, consider the following (oversimplified) snippet
//Sample taken from: https://learn.microsoft.com/en-us/dotnet/core/diagnostics/observability-otlp-example#3-create-an-api-endpoint
async Task<String> SendGreeting(ILogger<Program> logger)
{
// Create a new Activity scoped to the method
using var activity = greeterActivitySource.StartActivity("GreeterActivity");
var someObject = null;
// Add a tag to the Activity
activity.SetTag("greeting", "Hello World!"); //This will throw without listeners (No nullability check on activity)
activity?.SetTag("someProp", someObject.Prop1); //This will throw whith listeners (No nullability check on someObject)
return "Hello World!";
}
Solution
Run the test while listening to the activity sources and again while not
How to use
Install package
dotnet add package Contrib.Xunit.ActivityListenerTestFramework
Replace [Fact]
attributes with [ActivityCoverageFact]
and [Theory]
attributes with [ActivityCoverageTheory]
of the test that you want to get activity coverage, each accept a list of activity sources that will be sequencially listened on to enable coverage on the branching paths that are a result of activity nullability when there are no listeners, use an empty string (""
) to test with no listeners attached.
[ActivityCoverageFact("", nameof(SystemUnderTest), "RandomActitySource")] //Once with no listener, once listening to SystemUnderTest, once listening to RandomActivitySource
public void FactTest()
{
var subject = new SystemUnderTest();
var result = subject.TraceSome();
}
[ActivityCoverageFact] // Testing with an empty list of activity sources is also supported, this will prodcue equivalent behavior compared to the standard [Fact]
public void FactTest()
{
var subject = new SystemUnderTest();
var result = subject.TraceSome();
}
[Fact] //[Theory] These still works as expected and are not affected at all by the custom runner
public void RegularFactTest()
{
var subject = new SystemUnderTest();
var result = subject.TraceSome();
}
This will create a test case for each activity source, you may use an unbounded amount of sources, the test cases will be run while listening to each activity source in turn and you will achieve a higher amount of line and branch coverage.
[!Tip] Write your test as usual, don't focus on asserting the traceabilty
Limitations
This replaces the built-in assembly runner from xUnit, this means that you can not use this in conjuction with other packages/solutions that require a custom runner
Acknowlledgement
Special thanks to authors and mantainers of xUnit for their support.
Ideas taken from:
Product | Versions 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 is compatible. 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. |
-
.NETFramework 4.6.1
- System.Diagnostics.DiagnosticSource (>= 6.0.0)
- System.ValueTuple (>= 4.5.0)
- xunit.extensibility.execution (>= 2.8.0)
-
.NETStandard 2.0
- System.Diagnostics.DiagnosticSource (>= 6.0.0)
- System.ValueTuple (>= 4.5.0)
- xunit.extensibility.execution (>= 2.8.0)
NuGet packages
This package is not used by any NuGet packages.
GitHub repositories
This package is not used by any popular GitHub repositories.
Version | Downloads | Last updated |
---|---|---|
2.0.0 | 153 | 12/13/2024 |