XspecT 16.0.2-preview

This is a prerelease version of XspecT.
There is a newer version of this package available.
See the version list below for details.
dotnet add package XspecT --version 16.0.2-preview                
NuGet\Install-Package XspecT -Version 16.0.2-preview                
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="XspecT" Version="16.0.2-preview" />                
For projects that support PackageReference, copy this XML node into the project file to reference the package.
paket add XspecT --version 16.0.2-preview                
#r "nuget: XspecT, 16.0.2-preview"                
#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 XspecT as a Cake Addin
#addin nuget:?package=XspecT&version=16.0.2-preview&prerelease

// Install XspecT as a Cake Tool
#tool nuget:?package=XspecT&version=16.0.2-preview&prerelease                

XspecT: A fluent unit testing framework

Framework for writing and running automated tests in .Net in a fluent style, based on the popular "Given-When-Then" pattern, built upon XUnit, Moq, AutoMock, AutoFixture and FluentAssertions.

Whether you are beginner or expert in unit-testing, this framework will help you to write more descriptive, concise and maintainable tests.

Usage

It is assumed that you are already familiar with Xunit and Moq, or similar test- and mocking frameworks. There is an accompanying independent assertion framework called XspecT.Assert, which is built upon FluentAssertions, but with a less worthy syntax, based on the verbs Is, Has and Does instead of Should.

Is-assertions are recommended over Should-assertions for making the tests read more like specifications, listing requirements rather than asserting expected results.

This is an example of a complete test class (specification) with one test method (requirement):

using XspecT.Verification;
using XspecT.Fixture;

using static App.Calculator;

namespace App.Test;

public class CalculatorSpec : Spec<int>
{
    [Fact] public void WhenAdd_1_and_2_ThenSumIs_3() => When(_ => Add(1, 2)).Then().Result.Is(3);
}

When() is setting up the method to be called and Then() runs the test pipeline and provides the result.

Test a static method with [Theory]

If you are used to writing one test class per production class and use Theory for test input, you can use a similar style with XspecT. First you create your test-class overriding Spec<[ReturnType]> with the expected return type as generic argument. Then create a test-method, attributed with Theory and InlineData, called When[Something]. This method call When to setup the test pipeline with test data and the method to test. Finally verify the result by calling Then().Result (or only Result) on the returned pipeline and check the result with Is.

Example:

using XspecT.Verification;
using XspecT.Fixture;

using static App.Calculator;

namespace App.Test;

public class CalculatorSpec : Spec<int>
{
    [Theory]
    [InlineData(1, 1, 2)]
    [InlineData(3, 4, 7)]
    public void GivenTwoNumbers_WhenAdd_ReturnSum(int term1, int term2, int sum)
        => When(_ => Add(term1, term2)).Then().Result.Is(sum);

    [Theory]
    [InlineData(1, 1, 1)]
    [InlineData(3, 4, 12)]
    public void WhenMultiplyThenReturnProduct(int factor1, int factor2, int product)
        => When(_ => Multiply(factor1, factor2)).Then().Result.Is(product);
}

For more complex and realistic scenarios, it is recommended to create tests in a separate project from the production code, named [MyProject].Spec. The test-project should mimmic the production project's folder structure, but in addition have one folder for each class to test, named as the class. Within that folder, create one test-class per method to test.

Test a static void method

  • When testing a static void method, there is no return value to verify in result and by convention the generic TResult parameter should be set to object.
  • However you can use Throws or NotThrows to verify exceptions thrown.

Example:

namespace MyProject.Test.Validator;

public abstract class WhenVerifyAreEqual : Spec<object>
{
    protected WhenVerifyAreEqual() 
        => When(_ => MyProject.Validator.VerifyAreEqual(An<int>(), ASecond<int>()));

    public class Given_1_And_2 : WhenVerifyAreEqual
    {
        [Fact] public void ThenThrows_NotEqual() => Given(1, 2).Then().Throws<NotEqual>();
    }

    public class Given_2_And_2 : WhenVerifyAreEqual
    {
        [Fact] public void ThenDoNotThrow() => Given(2, 2).Then().DoesNotThrow();
    }
}

Test a class with dependencies

  • To test an instance method [MyClass].[MyMethod], create an abstract class named When[MyMethod] inheriting XspecT.Spec<[MyClass], [TheResult]>.

  • The subject under test will be created automatically with mocks and default values by AutoMock.

  • Subject-under-test is available as the single input parameter to the lambda that is provided to the method When You can supply or modify you own constructor arguments by calling Given or Given().Using.

  • To mock behaviour of any dependency call Given<[TheService]>().That(_ => _.[TheMethod](...)).Returns/Throws(...).

  • To verify a call to a dependency, write Then<[TheService]>([SomeLambdaExpression]).

  • Both mocking and verification of behaviour is based on Moq framework.

Example:

namespace MyProject.Spec.ShoppingService;

public abstract class WhenPlaceOrder : Spec<MyProject.ShoppingService, object>
{
    protected WhenPlaceOrder() 
        => When(_ => _.PlaceOrder(An<int>()))
        .Given<ICartRepository>().That(_ => _.GetCart(The<int>()))
        .Returns(() => A<Cart>(_ => _.Id = The<int>()));

    [Fact] public void ThenOrderIsCreated() => Then<IOrderService>(_ => _.CreateOrder(The<Cart>()));

    [Fact] public void ThenLogsOrderCreated()
        => Then<ILogger>(_ => _.Information($"OrderCreated from Cart {The<int>()}"));
}

All the examples above also works for async methods.

More examples and features can be found as Unit tests in the source code.

Product Compatible and additional computed target framework versions.
.NET net8.0 is compatible.  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. 
Compatible target framework(s)
Included target framework(s) (in package)
Learn more about Target Frameworks and .NET Standard.

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
17.2.0 35 1/18/2025
17.1.5 173 11/10/2024
17.1.4 93 11/10/2024
17.1.3 90 11/3/2024
17.1.2 89 11/2/2024
17.1.1 176 10/29/2024
17.1.0 101 10/28/2024
17.0.3 134 10/14/2024
17.0.2 94 10/14/2024
17.0.1 95 10/14/2024
17.0.0 99 10/12/2024
17.0.0-pre.3 54 10/8/2024
17.0.0-pre.2 48 10/6/2024
17.0.0-pre.1 60 10/6/2024
16.4.1 101 10/5/2024
16.4.0 119 10/4/2024
16.3.1 94 9/22/2024
16.3.0 97 9/22/2024
16.2.1 235 9/14/2024
16.2.0 105 9/14/2024
16.1.5 103 9/7/2024
16.1.4 109 9/7/2024
16.1.3 188 9/7/2024
16.1.2 97 9/6/2024
16.1.1 109 9/3/2024
16.1.0 107 9/2/2024
16.0.4 310 8/18/2024
16.0.4-preview 120 8/18/2024
16.0.3-preview 115 8/18/2024
16.0.2-preview 116 8/17/2024
16.0.1-preview 116 8/17/2024
16.0.0-preview 112 8/16/2024
15.7.0 201 8/7/2024
15.6.2 90 7/29/2024
15.6.1 199 7/14/2024
15.6.0 101 7/13/2024
15.5.4 164 7/7/2024
15.5.3 117 7/7/2024
15.5.2 112 7/7/2024
15.5.1 112 7/2/2024
15.5.0 116 6/30/2024
15.4.1 112 6/29/2024
15.4.0 133 6/24/2024
15.3.2 106 6/24/2024
15.3.1 122 6/23/2024
15.3.0 126 6/23/2024
15.2.1 132 6/20/2024
15.2.0 127 6/19/2024
15.1.3-preview 105 6/19/2024
15.1.2 123 6/18/2024
15.1.1 136 6/17/2024
15.1.0 136 6/16/2024
15.0.1 119 6/15/2024
15.0.0 121 6/9/2024
14.2.1 115 6/6/2024
14.2.0 110 6/6/2024
14.1.0 120 5/13/2024
14.0.0 111 5/9/2024
13.3.2 234 4/7/2024
13.3.1 124 1/31/2024
13.3.0 339 1/20/2024
13.2.3 129 1/15/2024
13.2.2 125 1/13/2024
13.2.1 133 1/2/2024
13.2.0 179 1/2/2024
13.1.2 140 12/19/2023
13.1.1 175 12/19/2023
13.1.0 142 12/18/2023
13.0.1 114 12/17/2023
13.0.0 127 12/17/2023
12.2.2 128 12/16/2023
12.2.1 123 12/16/2023
12.2.0 126 12/16/2023
12.1.1 133 12/16/2023
12.1.0 158 12/3/2023
12.0.0 144 12/2/2023
11.0.4 150 11/28/2023
11.0.3 245 11/19/2023
11.0.2 138 11/19/2023
11.0.1 144 11/18/2023
11.0.0 142 11/18/2023
10.0.2 151 11/18/2023
10.0.1 138 11/15/2023
10.0.0 145 11/12/2023
9.3.2 132 11/12/2023
9.3.1 131 11/12/2023
9.3.0 140 11/11/2023
9.2.1 144 11/11/2023
9.2.0 144 11/5/2023
9.1.1 149 10/29/2023
9.1.0 153 10/28/2023
9.0.0 162 10/28/2023
8.5.1 156 10/27/2023
8.5.0 155 10/26/2023
8.4.0 169 10/22/2023
8.3.1 170 10/22/2023
8.3.0 161 10/22/2023
8.2.1 156 10/22/2023
8.2.0 145 10/21/2023
8.1.2 153 10/21/2023
8.1.1 151 10/20/2023
8.1.0 139 10/20/2023
8.0.1 161 10/18/2023
8.0.0 144 10/16/2023
7.2.0 154 10/16/2023
7.1.1 144 10/12/2023
7.1.0 173 10/8/2023
7.0.1 145 10/1/2023
7.0.0 141 10/1/2023
6.4.0 164 9/30/2023
6.3.2 131 9/30/2023
6.3.1 156 9/30/2023
6.3.0 132 9/25/2023
6.2.4 153 9/15/2023
6.2.3 150 9/15/2023
6.2.2 136 9/15/2023
6.2.1 164 9/15/2023
6.2.0 151 9/14/2023
6.1.3 161 9/13/2023
6.1.2 175 9/12/2023
6.1.1 154 9/12/2023
6.1.0 184 9/10/2023
6.0.0 155 9/9/2023
5.5.0 167 9/8/2023
5.4.3 153 9/7/2023
5.4.2 175 9/5/2023
5.4.1 144 9/3/2023
5.4.0 243 8/28/2023
5.3.1 179 8/28/2023
5.3.0 157 8/27/2023
5.2.0 174 8/27/2023
5.1.1 172 8/26/2023
5.1.0 168 8/26/2023
5.0.0 177 8/26/2023
4.5.2 154 8/26/2023
4.5.1 160 8/26/2023
4.5.0 154 8/26/2023
4.4.7 162 8/22/2023
4.4.6 149 8/22/2023
4.4.5 144 8/21/2023
4.4.4 176 8/20/2023
4.4.3 170 8/16/2023
4.4.2 180 8/15/2023
4.4.1 177 8/15/2023
4.4.0 195 8/15/2023
4.3.1 176 8/14/2023
4.3.0 195 8/14/2023
4.2.0 185 8/14/2023
4.1.1 176 8/11/2023
4.1.0 174 8/9/2023
4.0.0 190 8/8/2023
3.3.2 173 8/7/2023
3.3.1 169 8/6/2023
3.3.0 176 8/6/2023
3.2.1 171 8/6/2023
3.2.0 208 8/6/2023
3.1.0 204 8/5/2023
3.0.0 191 8/2/2023
2.4.1 188 8/1/2023
2.4.0 176 8/1/2023
2.3.1 182 7/30/2023
2.3.0 172 7/30/2023
2.2.3 174 7/29/2023
2.2.2 184 7/28/2023
2.2.1 180 7/24/2023
2.2.0 189 7/24/2023
2.1.1 189 7/23/2023
2.1.0 186 7/23/2023
2.0.1 188 7/21/2023
2.0.0 196 7/21/2023
1.1.0 206 7/20/2023
1.0.0 173 7/20/2023

Handle line breaks in specification