tsp-csharp-xunit

star 224

Comprehensive xUnit testing guide for C# projects. Use when writing unit tests, setting up test projects, creating data-driven tests, mocking dependencies, or organizing test suites.

querylenshq By querylenshq schedule Updated 6/7/2026

name: tsp-csharp-xunit description: "Comprehensive xUnit testing guide for C# projects. Use when writing unit tests, setting up test projects, creating data-driven tests, mocking dependencies, or organizing test suites."

xUnit Testing — TSP Standards

Project Setup

  • Test project name: {ProjectName}.Tests
  • Required packages: Microsoft.NET.Test.Sdk, xunit, xunit.runner.visualstudio
  • Recommended: FluentAssertions, NSubstitute (or Moq)
  • Run tests: dotnet test (use --filter for targeted runs)

Test Structure

  • No test class attribute required (unlike MSTest/NUnit)
  • Use constructor for per-test setup; implement IDisposable for teardown
  • Use IClassFixture<T> for shared context within a test class
  • Use ICollectionFixture<T> for shared context across test classes
  • Use IAsyncLifetime for async setup/teardown
public class OrderServiceTests : IClassFixture<DatabaseFixture>, IDisposable
{
    private readonly DatabaseFixture _db;

    public OrderServiceTests(DatabaseFixture db)
    {
        _db = db;
    }

    [Fact]
    public void CreateOrder_WithValidItems_ReturnsOrderId()
    {
        // Arrange
        var service = new OrderService(_db.Context);
        var items = new[] { new OrderItem("SKU-1", 2) };

        // Act
        var result = service.CreateOrder(items);

        // Assert
        result.Should().BeGreaterThan(0);
    }

    public void Dispose() { /* cleanup */ }
}

Naming Conventions

  • Test class: {ClassUnderTest}Tests
  • Test method: {Method}_{Scenario}_{ExpectedResult}
  • Examples:
    • GetUser_WhenIdIsInvalid_ReturnsNotFound
    • CalculateTotal_WithDiscount_AppliesPercentage
    • SaveAsync_WhenCancelled_ThrowsOperationCancelled

Data-Driven Tests

  • [Theory] + [InlineData] for simple inline values
  • [Theory] + [MemberData] for method/property-based data
  • [Theory] + [ClassData] for reusable data generators
  • Use meaningful parameter names that describe the scenario
[Theory]
[InlineData(0, false)]
[InlineData(5, true)]
[InlineData(-1, false)]
public void IsValidQuantity_ReturnsExpected(int quantity, bool expected)
{
    var result = OrderValidator.IsValidQuantity(quantity);

    result.Should().Be(expected);
}

Assertions

  • Prefer FluentAssertions when available:
    • result.Should().Be(expected)
    • collection.Should().HaveCount(3).And.Contain(x => x.IsActive)
    • act.Should().ThrowAsync<InvalidOperationException>()
  • Built-in xUnit assertions as fallback:
    • Assert.Equal, Assert.True, Assert.Throws<T>
    • Assert.Contains / Assert.DoesNotContain for collections
  • One logical behavior per test — multiple assertions are fine if testing one concept

Mocking

  • Prefer NSubstitute for readability; Moq is also acceptable
  • Mock interfaces, not concrete classes
  • Verify interactions only when the interaction IS the behavior being tested
  • Avoid over-mocking — if you're mocking more than 3 dependencies, the class may need refactoring
var repo = Substitute.For<IOrderRepository>();
repo.GetByIdAsync(42, Arg.Any<CancellationToken>())
    .Returns(new Order { Id = 42 });

var service = new OrderService(repo);
var result = await service.GetOrderAsync(42, CancellationToken.None);

result.Should().NotBeNull();
result.Id.Should().Be(42);

Test Organization

  • Group tests by feature or component
  • Use [Trait("Category", "Integration")] for categorization
  • Skip tests with Skip = "reason" when conditionally disabled
  • Use ITestOutputHelper for diagnostic output (not Console.WriteLine)

Rules

  • Tests must run in any order and in parallel — no shared mutable state
  • Avoid disk I/O — use in-memory alternatives or mocks
  • No Thread.Sleep — use Task.Delay with cancellation tokens in async tests
  • Every new or changed public API must have corresponding tests
Install via CLI
npx skills add https://github.com/querylenshq/ef-querylens --skill tsp-csharp-xunit
Repository Details
star Stars 224
call_split Forks 6
navigation Branch main
article Path SKILL.md
More from Creator