Introduction
Getting started
Fluent Assertions is a set of .NET extension methods that allow you to more naturally specify the expected outcome of a TDD or BDD-style unit test. This enables a simple intuitive syntax that all starts with the following using
statement:
using FluentAssertions;
This brings a lot of extension methods into the current scope. For example, to verify that a string begins, ends and contains a particular phrase.
string actual = "ABCDEFGHI";
actual.Should().StartWith("AB").And.EndWith("HI").And.Contain("EF").And.HaveLength(9);
To verify that all elements of a collection match a predicate and that it contains a specified number of elements.
IEnumerable<int> numbers = new[] { 1, 2, 3 };
numbers.Should().OnlyContain(n => n > 0);
numbers.Should().HaveCount(4, "because we thought we put four items in the collection");
The nice thing about the second failing example is that it will throw an exception with the message (and notice that it uses the name of the variable numbers
):
Expected numbers to contain 4 item(s) because we thought we put four items in the collection, but found 3.
To verify that a particular business rule is enforced using exceptions:
var recipe = new RecipeBuilder()
.With(new IngredientBuilder().For("Milk").WithQuantity(200, Unit.Milliliters))
.Build();
var action = () => recipe.AddIngredient("Milk", 100, Unit.Spoon);
action
.Should().Throw<RuleViolationException>()
.WithMessage("*change the unit of an existing ingredient*")
.And.Violations.Should().Contain(BusinessRule.CannotChangeIngredientQuantity);
One neat feature is the ability to chain a specific assertion on top of an assertion that acts on a collection or graph of objects.
dictionary.Should().ContainValue(myClass).Which.SomeProperty.Should().BeGreaterThan(0);
someObject.Should().BeOfType<Exception>().Which.Message.Should().Be("Other Message");
xDocument.Should().HaveElement("child").Which.Should().BeOfType<XElement>().And.HaveAttribute("attr", "1");
If the first one fails, you will get a message like:
Expected dictionary["key"].SomeProperty to be greater than 0, but found -2
This chaining can make your unit tests a lot easier to read.
Global Configurations
Fluent Assertions’ AssertionConfiguration
has several methods and properties that can be used to change the way it executes assertions or the defaults it will use for comparing object graphs. Changing those settings at the right time can be difficult, depending on the test framework. That’s why Fluent Assertions offers a special assembly-level attribute that can be used to have some code executed before the first assertion is executed. It will be called only once per test run, but you can use the attribute multiple times.
[assembly: AssertionEngineInitializer(typeof(Initializer), nameof(Initializer.Initialize))]
public static class Initializer
{
public static void Initialize()
{
AssertionConfiguration.Current.Equivalency.Modify(options => options
.ComparingByValue<DirectoryInfo>());
}
}
Detecting Test Frameworks
Fluent Assertions supports a lot of different unit testing frameworks. Just add a reference to the corresponding test framework assembly to the unit test project. Fluent Assertions will automatically find the corresponding assembly and use it for throwing the framework-specific exceptions.
If, for some unknown reason, Fluent Assertions fails to find the assembly, and you’re running under .NET 4.7 or a .NET 6.0 project, try specifying the framework explicitly using a configuration setting in the project’s app.config. If it cannot find any of the supported frameworks, it will fall back to using a custom AssertionFailedException
exception class.
<configuration>
<appSettings>
<!-- Supported values: nunit, xunit2, xunit3, mstestv2, mspec and tunit -->
<add key="FluentAssertions.TestFramework" value="nunit"/>
</appSettings>
</configuration>
Just add NuGet package “FluentAssertions” to your test project.
Subject Identification
Fluent Assertions can use the C# code of the unit test to extract the name of the subject and use that in the assertion failure. Consider for instance this statement:
string username = "dennis";
username.Should().Be("jonas");
This will throw a test framework-specific exception with the following message:
Expected username to be "jonas" with a length of 5, but "dennis" has a length of 6, differs near "den" (index 0).
The way this works is that Fluent Assertions will try to traverse the current stack trace to find the line and column numbers as well as the full path to the source file. Since it needs the debug symbols for that, this will require you to compile the unit test projects in debug mode, even on your build servers.
Also, this does not work with PathMap
for unit test projects as it assumes that source files are present on the path returned from StackFrame.GetFileName()
.
Now, if you’ve built your own extensions that use Fluent Assertions directly, you can tell it to skip that extension code while traversing the stack trace. Consider for example the customer assertion:
public class CustomerAssertions
{
private readonly Customer customer;
public CustomerAssertions(Customer customer)
{
this.customer = customer;
}
[CustomAssertion]
public void BeActive(string because = "", params object[] becauseArgs)
{
customer.Active.Should().BeTrue(because, becauseArgs);
}
}
And it’s usage:
myClient.Should().BeActive("because we don't work with old clients");
Without the [CustomAssertion]
attribute, Fluent Assertions would find the line that calls Should().BeTrue()
and treat the customer
variable as the subject-under-test (SUT). But by applying this attribute, it will ignore this invocation and instead find the SUT by looking for a call to Should().BeActive()
and use the myClient
variable instead.
Alternatively, you can add the [assembly:CustomAssertionsAssembly]
attribute to a file within the project to tell Fluent Assertions that all code in that assembly should be treated as custom assertion code.
Assertion Scopes
You can batch multiple assertions into an AssertionScope
so that Fluent Assertions throws one exception at the end of the scope with all failures.
E.g.
using (new AssertionScope())
{
5.Should().Be(10);
"Actual".Should().Be("Expected");
}
The above will batch the two failures, and throw an exception at the point of disposing the AssertionScope
displaying both errors.
E.g. Exception thrown at point of dispose contains:
Expected value to be 100, but found 95 (difference of -5).
Expected string to be "Expected" with a length of 8, but "Actual" has a length of 6, differs near "Act" (index 0).
You can even nest two of those scopes and give them suitable names:
using var outerScope = new AssertionScope("Test1");
using var innerScope = new AssertionScope("Test2");
nonEmptyList.Should().BeEmpty();
This will give you:
Expected Test1/Test2/nonEmptyList to be empty, but found at least one item {1}.
In more sophisticated scenarios, you might want to intercept the assertion raised within an AssertionScope
and prevent it from throwing an exception.
using (var scope = new AssertionScope())
{
5.Should().Be(10);
// other assertion left out for brevity...
// Collect all the failure messages that occurred up to this point
string[] failures = scope.Discard();
// The closing brace will not throw any exceptions anymore
}
For more examples take a look at the AssertionScopeSpecs.cs in Unit Tests.
Scoped IValueFormatter
s
You can add a custom value formatter inside a scope to selectively customize formatting of an object based on the context of the test. To achieve that, you can do following:
using var scope = new AssertionScope();
var formatter = new CustomFormatter();
scope.FormattingOptions.AddFormatter(formatter);
You can even add formatters to nested assertion scopes and the nested scope will pick up all previously defined formatters:
using var outerScope = new AssertionScope();
var outerFormatter = new OuterFormatter();
var innerFormatter = new InnerFormatter();
outerScope.FormattingOptions.AddFormatter(outerFormatter);
using var innerScope = new AssertionScope();
innerScope.FormattingOptions.AddFormatter(innerFormatter);
// At this point outerFormatter and innerFormatter will be available
Note: If you modify the scoped formatters inside the nested scope, it won’t touch the scoped formatters from the outer scope:
using var outerScope = new AssertionScope();
var outerFormatter = new OuterFormatter();
var innerFormatter = new InnerFormatter();
outerScope.FormattingOptions.AddFormatter(outerFormatter);
using (var innerScope = new AssertionScope())
{
innerScope.FormattingOptions.AddFormatter(innerFormatter);
innerScope.FormattingOptions.RemoveFormatter(outerFormatter);
// innerScope only contains innerFormatter
}
// outerScope still contains outerFormatter
Licensing
Version 7 will remain fully open-source indefinitely and receive bugfixes and other important corrections.
Versions 8 and beyond are/will be free for open-source projects and non-commercial use, but commercial use requires a paid license. Check out the license page for more information.
Since Fluent Assertions 8 doesn’t need any license key, there’s a soft warning that is displayed for every test run. This is to remind consumers that you need a paid license for commercial use. To suppress this warning, there’s a static property called License.Accepted
that can be set to true
. You can add the following code to your test project to automatically toggle this flag.
[assembly: FluentAssertions.Extensibility.AssertionEngineInitializer(
typeof(AssertionEngineInitializer),
nameof(AssertionEngineInitializer.AcknowledgeSoftWarning))]
public static class AssertionEngineInitializer
{
public static void AcknowledgeSoftWarning()
{
License.Accepted = true;
}
}