SMock breaks down the barriers of testing legacy code, third-party dependencies, and static APIs. Built on MonoMod runtime modification technology, SMock gives you the power to mock what others can't.
- Mock Static Methods: The only .NET library that handles static methods seamlessly
- Two API Styles: Choose Hierarchical (with validation) or Sequential (disposable) patterns
- Zero Configuration: Works with your existing test frameworks (NUnit, xUnit, MSTest)
- Complete Feature Set: Async/await, parameter matching, callbacks, exceptions, unsafe code
Install-Package SMockdotnet add package SMock💡 Pro Tip: SMock works great with any testing framework - NUnit, xUnit, MSTest, you name it!
// Mock a static method in just one line - Sequential API
using var mock = Mock.Setup(() => File.ReadAllText("config.json"))
.Returns("{ \"setting\": \"test\" }");
// Your code now uses the mocked value!
var content = File.ReadAllText("config.json"); // Returns test JSON
// Or use the Hierarchical API with inline validation
Mock.Setup(() => DateTime.Now, () =>
{
var result = DateTime.Now;
Assert.AreEqual(new DateTime(2024, 1, 1), result);
}).Returns(new DateTime(2024, 1, 1));SMock uses MonoMod to create runtime hooks that intercept method calls:
- Non-Invasive: No source code changes required
- Isolated: Each test runs in isolation
- Fast: Minimal performance overhead
- Auto-Cleanup: Hooks automatically removed after test completion
// 1. Setup: Create a mock for the target method
var mock = Mock.Setup(() => DateTime.Now);
// 2. Configure: Define return values or behaviors
mock.Returns(new DateTime(2024, 1, 1));
// 3. Execute: Run your code - calls are intercepted
var now = DateTime.Now; // Returns mocked value
// 4. Cleanup: Dispose mock (Sequential) or automatic (Hierarchical)
mock.Dispose(); // Or automatic with 'using'SMock provides two distinct API patterns to fit different testing preferences:
Perfect for clean, scoped mocking with automatic cleanup:
[Test]
public void TestFileOperations()
{
// Mock file existence check
using var existsMock = Mock.Setup(() => File.Exists("test.txt"))
.Returns(true);
// Mock file content reading
using var readMock = Mock.Setup(() => File.ReadAllText("test.txt"))
.Returns("Hello World");
// Your code under test
var processor = new FileProcessor();
var result = processor.ProcessFile("test.txt");
Assert.AreEqual("HELLO WORLD", result);
} // Mocks automatically cleaned up herePerfect for inline validation during mock execution:
[Test]
public void TestDatabaseConnection()
{
var expectedConnectionString = "Server=localhost;Database=test;";
Mock.Setup(context => DatabaseConnection.Connect(It.IsAny<string>()), () =>
{
// This validation runs DURING the mock execution
var actualCall = DatabaseConnection.Connect(expectedConnectionString);
Assert.IsNotNull(actualCall);
Assert.IsTrue(actualCall.IsConnected);
}).Returns(new MockConnection { IsConnected = true });
// Test your service
var service = new DatabaseService();
service.InitializeConnection(expectedConnectionString);
}SMock is designed for minimal performance impact:
- Runtime Hooks: Only active during tests
- Zero Production Overhead: No dependencies in production builds
- Efficient Interception: Built on MonoMod's optimized IL modification
- Benchmarked: Comprehensive performance testing with BenchmarkDotNet
| Operation | Overhead | Notes |
|---|---|---|
| Mock Setup | ~1-2ms | One-time cost per mock |
| Method Interception | <0.1ms | Minimal runtime impact |
| Cleanup | <1ms | Automatic hook removal |
| Memory Usage | Minimal | Temporary IL modifications only |
If your mocks are not being applied and you're getting the original method behavior instead of the mocked behavior, this is likely due to compiler optimizations. The compiler may inline or optimize method calls, preventing SMock from intercepting them.
Solutions:
-
Run tests in Debug configuration:
dotnet test --configuration Debug -
Disable compiler optimization in your test project by adding this to your
.csproj:<PropertyGroup> <Optimize>false</Optimize> </PropertyGroup>
-
Disable optimization for specific methods using the
MethodImplattribute:[Test] [MethodImpl(MethodImplOptions.NoOptimization)] public void MyTestMethod() { using var mock = Mock.Setup(() => File.ReadAllText("config.json")) .Returns("{ \"setting\": \"test\" }"); // Your test code here }
This issue typically occurs in Release builds where the compiler aggressively optimizes method calls. Using any of the above solutions will ensure your mocks work correctly.
This library is available under the MIT license.
⚡ Get Started Now | 📚 View Examples | 💬 Join Discussion
Made with ❤️ by @SvetlovA and the SMock community