A set of extension methods for mocking HttpClient and IHttpClientFactory with Moq.
Mocking HttpClient directly is notoriously difficult; the general approach has been to either create a wrapper of some form to mock instead or use a specific testing library. However, the former is typically undesirable, and the latter requires switching to a separate mocking system for HTTP calls which may be less flexible or awkward in conjunction with other mocks. These extensions instead allow HttpClient to be mocked the same way as everything else using Moq without excessive boilerplate.
Install-Package Moq.Contrib.HttpClient
or dotnet add package Moq.Contrib.HttpClient
The library adds request/response versions of the standard Moq methods:
- Setup → SetupRequest, SetupAnyRequest
- SetupSequence → SetupRequestSequence, SetupAnyRequestSequence
- Verify → VerifyRequest, VerifyAnyRequest
- Returns(Async) → ReturnsResponse
InSequence().Setup()
is supported as well.
All Setup and Verify helpers have the same overloads, abbreviated here:
SetupAnyRequest()
SetupRequest([HttpMethod method, ]Predicate<HttpRequestMessage> match)
SetupRequest(string|Uri requestUrl[, Predicate<HttpRequestMessage> match])
SetupRequest(HttpMethod method, string|Uri requestUrl[, Predicate<HttpRequestMessage> match])
The match
predicate allows for more intricate matching, including headers, and may be async to inspect the request body.
The response helpers simplify sending a StringContent, ByteArrayContent, StreamContent, or just a status code. All overloads take an action to further configure the response (i.e. setting headers).
ReturnsResponse(HttpStatusCode statusCode[, HttpContent content], Action<HttpResponseMessage> configure = null)
ReturnsResponse([HttpStatusCode statusCode, ]string content, string mediaType = null, Encoding encoding = null, Action<HttpResponseMessage> configure = null))
ReturnsResponse([HttpStatusCode statusCode, ]byte[]|Stream content, string mediaType = null, Action<HttpResponseMessage> configure = null)
// All requests made with HttpClient go through its handler's SendAsync() which we mock
var handler = new Mock<HttpMessageHandler>();
var client = handler.CreateClient();
// A simple example that returns 404 for any request
handler.SetupAnyRequest()
.ReturnsResponse(HttpStatusCode.NotFound);
// Match GET requests to an endpoint that returns json (defaults to 200 OK)
handler.SetupRequest(HttpMethod.Get, "https://example.com/api/stuff")
.ReturnsResponse(JsonConvert.SerializeObject(model), "application/json");
// Setting additional headers on the response using the optional configure action
handler.SetupRequest("https://example.com/api/stuff")
.ReturnsResponse(bytes, configure: response =>
{
response.Content.Headers.LastModified = new DateTime(2018, 3, 9);
})
.Verifiable(); // Naturally we can use Moq methods as well
// Verify methods are provided matching the setup helpers
handler.VerifyAnyRequest(Times.Exactly(3));
// The request helpers can take a predicate for more intricate request matching
handler.SetupRequest(r => r.Headers.Authorization?.Parameter != authToken)
.ReturnsResponse(HttpStatusCode.Unauthorized);
// This can be async as well to inspect the request body
handler
.SetupRequest(HttpMethod.Post, url, async request =>
{
// This setup will only match calls with the expected id
var json = await request.Content.ReadAsStringAsync();
var model = JsonConvert.DeserializeObject<Model>();
return model.Id == expected.Id;
})
.ReturnsResponse(HttpStatusCode.Created);
// We can also use this to check for a specific query param without regard for any others
handler.SetupRequest(r => ((Url) r.RequestUri).QueryParams["foo"].Equals("bar"))
.ReturnsResponse("stuff");
The last example uses Flurl, a fluent URL builder, to assist in checking the query string. This can also make constructing a URL easier, for instance if we wanted to match an exact URL and query string. See the request extensions tests for an example.
Moq has two types of sequences:
SetupSequence()
which creates one setup that returns values in sequence, andInSequence().Setup()
which creates multiple setups underWhen()
conditions to ensure that they only match in order.
The latter can be useful for cases where separate requests independent of each other must be made in a certain order; their setups can be defined in a sequence such that one must match before the other. This is similar to other testing libraries that work by queueing responses.
See the sequence extensions tests for examples of each.
Since it's Moq, we can use the normal Returns method together with the request helpers for more complex responses:
handler.SetupRequest("https://example.com/hello")
.Returns(async (HttpRequestMessage request, CancellationToken cancellationToken) =>
new HttpResponseMessage()
{
Content = new StringContent($"Hello, {await request.Content.ReadAsStringAsync()}")
});
var response = await client.PostAsync("https://example.com/hello", new StringContent("world"));
var body = await response.Content.ReadAsStringAsync(); // Hello, world
It is common to see HttpClient wrapped in a using
since it's IDisposable, but this is actually incorrect and can lead to your application eating up sockets. The standard advice is to keep a static or singleton HttpClient for the lifetime of the application, but this has the drawback of not responding to DNS changes.
ASP.NET Core introduces a new IHttpClientFactory which "manages the pooling and lifetime of underlying HttpClientMessageHandler instances to avoid common DNS problems that occur when manually managing HttpClient lifetimes." As a bonus, it also makes more accessible a little-known feature in HttpClient: the ability to plug in middleware (for example, using Polly to automatically handle retries and failures).
Depending on the usage, your constructors may simply take an HttpClient injected via IHttpClientFactory. If the constructor takes the factory itself instead, this can be mocked the same way:
var handler = new Mock<HttpMessageHandler>();
var factory = handler.CreateClientFactory();
// Named clients can be configured as well (overriding the default)
Mock.Get(factory).Setup(x => x.CreateClient("api"))
.Returns(() =>
{
var client = handler.CreateClient();
client.BaseAddress = ApiBaseUrl;
return client;
});
The factory can then be passed into the class or injected via AutoMocker, and code calling factory.CreateClient()
will receive clients backed by the mock handler.
Though it may be a faux pas to point to the unit tests as documentation, in this case the library is specifically for testing, and so they were written with this in mind. Thus, for some more complete working examples (with comments), please see here:
- Request extensions tests — these strictly cover the Setup & Verify helpers
- Response extensions tests — these focus on the ReturnsResponse helpers
- Sequence extensions tests — these demonstrate mocking sequences, as mentioned above
MIT license.
If you found this useful, please consider buying me a coffee cup of tea!