Visit the official documentation website for comprehensive guides, API reference, and examples.
- Quick Start - Get started in 5 minutes
- Usage Guide - Learn core features
- Advanced Features - Custom matchers, assertions, and more
- Building - Build from source
- Contributing - Contribution guidelines
Mockly is a powerful and flexible HTTP mocking library for .NET that makes it easy to test code that depends on HttpClient. It provides a fluent API for configuring HTTP request mocks, capturing request details, and asserting on HTTP interactions in your tests.
The library supports:
- .NET Framework 4.7.2 and higher
- .NET 8.0 and higher
- FluentAssertions 7.x and 8.x integration for expressive test assertions
Unlike other HTTP mocking libraries, Mockly offers:
- Fluent, intuitive API - Chain method calls to build complex mocking scenarios with ease
- Wildcard pattern matching - Match URLs using wildcards (
*) in paths and query strings - Custom matchers - Use predicates for advanced request matching logic
- Request capture & inspection - Automatically capture all requests with full metadata (headers, body, timestamp)
- Powerful assertions - Built-in FluentAssertions extensions for verifying HTTP behavior
- Diagnostic support - Detailed error messages when unexpected requests occur
- Extensibility - Design allows for custom response generators and matchers
- Zero configuration - Works out of the box with sensible defaults
- Performance optimized - Regex patterns are cached for efficient matching
- Invocation limits - Restrict how many times a mock can respond using
Once(),Twice(), orTimes(n)
Mockly is created and maintained by Dennis Doomen, also the creator of FluentAssertions, PackageGuard, Reflectify, Pathy and the .NET Library Starter Kit. It's designed to work seamlessly with modern .NET testing practices and integrates naturally with FluentAssertions for expressive test assertions.
mock.ForGet().WithPath("/api/users/*").RespondsWithJsonContent(user);
mock.ForPost().WithPath("/api/data").WithQuery("?filter=*").RespondsWithStatus(HttpStatusCode.Created);When an unexpected request occurs and there are configured mocks, Mockly helps you diagnose by reporting the closest matching mock (method, scheme/host/path/query) so you can quickly see what to adjust in your setup.
Unexpected request to:
GET http://localhost/fnv_collectiveschemes(111)
Closest matching mock:
GET https://*/fnv_collectiveschemes(123*)
Registered mocks:
- GET https://*/fnv_collectiveschemes
- POST https://*/fnv_collectiveschemes
- GET https://*/fnv_collectiveschemes(123*)
- GET https://*/fnv_collectiveschemes(123*) (1 custom matcher(s)) where (request => request.Uri?.Query == "?$count=1")
- GET https://*/fnv_collectiveschemes(456)
var patches = new RequestCollection();
mock.ForPatch().WithPath("/api/update").CollectingRequestIn(patches);
// After test execution
patches.Count.Should().Be(3);
patches.First().Path.Should().Contain("/api/update");mock.Should().HaveAllRequestsCalled();
mock.Requests.Should().NotBeEmpty();
mock.Requests.Should().NotContainUnexpectedCalls();
// Assert JSON-equivalence using a JSON string (ignores formatting/ordering)
mock.Requests.Should().ContainRequest()
.WithBodyMatchingJson("{ \"id\": 1, \"name\": \"x\" }");
// Assert the body deserializes and is equivalent to an object graph
var expected = new { id = 1, name = "x" };
mock.Requests.Should().ContainRequestForUrl("http://localhost:7021/api/*")
.WithBodyEquivalentTo(expected);- JSON content with automatic serialization
- Test data builder integration via
IResponseBuilder<T> - Raw string content
- Custom HTTP status codes
- Custom response generators
- OData support
mock.FailOnUnexpectedCalls = true; // Default behavior
// Throws UnexpectedRequestException if an unmocked request is madeInstall the package:
dotnet add package mocklyTo get the assertions, also install one of the two assertion packages, depending on which version of FluentAssertions you're using:
dotnet add package FluentAssertions.Mockly.v7
dotnet add package FluentAssertions.Mockly.v8Basic usage:
using Mockly;
using FluentAssertions;
// Arrange
var mock = new HttpMock();
mock.ForGet()
.WithPath("/api/users/123")
.RespondsWithJsonContent(new { Id = 123, Name = "John Doe" });
HttpClient client = mock.GetClient();
// Act
// Note: BaseAddress defaults to https://localhost/
var response = await client.GetAsync("/api/users/123");
var content = await response.Content.ReadAsStringAsync();
// Assert
response.StatusCode.Should().Be(HttpStatusCode.OK);
content.Should().Contain("John Doe");
mock.Should().HaveAllRequestsCalled();For complete documentation and advanced examples, visit dennisdoomen.github.io/mockly
To build this repository locally, you need:
- The .NET SDKs for .NET 4.7 and 8.0.
- Visual Studio, JetBrains Rider or Visual Studio Code with the C# DevKit
Build using PowerShell:
./build.ps1Or with the Nuke tool:
nukeFor more details, see the Building documentation.
Your contributions are always welcome! Please have a look at the contribution guidelines first.
For detailed contribution information, visit the Contributing documentation.
Previous contributors:
(Made with contrib.rocks)
This library uses Semantic Versioning to give meaning to the version numbers. For the versions available, see the releases on this repository.
This library wouldn't have been possible without the following tools, packages and companies:
- FluentAssertions - Fluent API for asserting the results of unit tests by Dennis Doomen
- Nuke - Smart automation for DevOps teams and CI/CD pipelines by Matthias Koch
- xUnit - Community-focused unit testing tool for .NET by Brad Wilson
- Coverlet - Cross platform code coverage for .NET by Toni Solarin-Sodara
- GitVersion - From git log to SemVer in no time
- ReportGenerator - Converts coverage reports by Daniel Palme
- StyleCopyAnalyzer - StyleCop rules for .NET
- Roslynator - A set of code analysis tools for C# by Josef Pihrt
- CSharpCodingGuidelines - Roslyn analyzers by Bart Koelman to go with the C# Coding Guidelines
- Meziantou - Another set of awesome Roslyn analyzers by Gérald Barré
You may also be interested in:
- FluentAssertions - The assertion library that Mockly integrates with
- PackageGuard - Get a grip on your open-source packages
- Reflectify - Reflection extensions without causing dependency pains
- Pathy - Fluently building and using file and directory paths without binary dependencies
- .NET Library Starter Kit - A battle-tested starter kit for building open-source and internal NuGet libraries
This project is licensed under the MIT License - see the LICENSE file for details.
