SKILL.md
TUnit Best Practices
Your goal is to help me write effective unit tests with TUnit, covering both standard and data-driven testing approaches.
Project Setup
- Use a separate test project with naming convention
[ProjectName].Tests
- Reference TUnit package and TUnit.Assertions for fluent assertions
- Create test classes that match the classes being tested (e.g.,
CalculatorTestsforCalculator)
- Use .NET SDK test commands:
dotnet testfor running tests
- TUnit requires .NET 8.0 or higher
Test Structure
- No test class attributes required (like xUnit/NUnit)
- Use
[Test]attribute for test methods (not[Fact]like xUnit)
- Follow the Arrange-Act-Assert (AAA) pattern
- Name tests using the pattern
MethodName_Scenario_ExpectedBehavior
- Use lifecycle hooks:
[Before(Test)]for setup and[After(Test)]for teardown
- Use
[Before(Class)]and[After(Class)]for shared context between tests in a class
- Use
[Before(Assembly)]and[After(Assembly)]for shared context across test classes
- TUnit supports advanced lifecycle hooks like
[Before(TestSession)]and[After(TestSession)]
Standard Tests
- Keep tests focused on a single behavior
- Avoid testing multiple behaviors in one test method
- Use TUnit's fluent assertion syntax with
await Assert.That()
- Include only the assertions needed to verify the test case
- Make tests independent and idempotent (can run in any order)
- Avoid test interdependencies (use
[DependsOn]attribute if needed)
Data-Driven Tests
- Use
[Arguments]attribute for inline test data (equivalent to xUnit's[InlineData])
- Use
[MethodData]for method-based test data (equivalent to xUnit's[MemberData])
- Use
[ClassData]for class-based test data
- Create custom data sources by implementing
ITestDataSource
- Use meaningful parameter names in data-driven tests
- Multiple
[Arguments]attributes can be applied to the same test method
Assertions
- Use
await Assert.That(value).IsEqualTo(expected)for value equality
- Use
await Assert.That(value).IsSameReferenceAs(expected)for reference equality
- Use
await Assert.That(value).IsTrue()orawait Assert.That(value).IsFalse()for boolean conditions
- Use
await Assert.That(collection).Contains(item)orawait Assert.That(collection).DoesNotContain(item)for collections
- Use
await Assert.That(value).Matches(pattern)for regex pattern matching
- Use
await Assert.That(action).Throws<TException>()orawait Assert.That(asyncAction).ThrowsAsync<TException>()to test exceptions
- Chain assertions with
.Andoperator:await Assert.That(value).IsNotNull().And.IsEqualTo(expected)
- Use
.Oroperator for alternative conditions:await Assert.That(value).IsEqualTo(1).Or.IsEqualTo(2)
- Use
.Within(tolerance)for DateTime and numeric comparisons with tolerance
- All assertions are asynchronous and must be awaited
Advanced Features
- Use
[Repeat(n)]to repeat tests multiple times
- Use
[Retry(n)]for automatic retry on failure
- Use
[ParallelLimit<T>]to control parallel execution limits
- Use
[Skip("reason")]to skip tests conditionally
- Use
[DependsOn(nameof(OtherTest))]to create test dependencies
- Use
[Timeout(milliseconds)]to set test timeouts
- Create custom attributes by extending TUnit's base attributes
Test Organization
- Group tests by feature or component
- Use
[Category("CategoryName")]for test categorization
- Use
[DisplayName("Custom Test Name")]for custom test names
- Consider using
TestContextfor test diagnostics and information
- Use conditional attributes like custom
[WindowsOnly]for platform-specific tests
Performance and Parallel Execution
- TUnit runs tests in parallel by default (unlike xUnit which requires explicit configuration)
- Use
[NotInParallel]to disable parallel execution for specific tests
- Use
[ParallelLimit<T>]with custom limit classes to control concurrency
- Tests within the same class run sequentially by default
- Use
[Repeat(n)]with[ParallelLimit<T>]for load testing scenarios
Migration from xUnit
- Replace
[Fact]with[Test]
- Replace
[Theory]with[Test]and use[Arguments]for data
- Replace
[InlineData]with[Arguments]
- Replace
[MemberData]with[MethodData]
- Replace
Assert.Equalwithawait Assert.That(actual).IsEqualTo(expected)
- Replace
Assert.Truewithawait Assert.That(condition).IsTrue()
- Replace
Assert.Throws<T>withawait Assert.That(action).Throws<T>()
- Replace constructor/IDisposable with
[Before(Test)]/[After(Test)]
- Replace
IClassFixture<T>with[Before(Class)]/[After(Class)]
Why TUnit over xUnit?
TUnit offers a modern, fast, and flexible testing experience with advanced features not present in xUnit, such as asynchronous assertions, more refined lifecycle hooks, and improved data-driven testing capabilities. TUnit's fluent assertions provide clearer and more expressive test validation, making it especially suitable for complex .NET projects.