Snapshot Testing with C# and WireMock:

How to stop mocking APIs manually and start automating your tests like a wizard 🧙

Featured image

What are we trying to solve

Picture this: Some time ago I had situation in which I had to test service that aggregated data from dozens of endpoints in multiple services (please don’t judge me, life is not always like we would like it to be 😭). It looked something like:

arch1

Implementation is one thing, but then you are going to face another wall - how to integrate test it, without spending years on mocking data for each case and each service call.

I needed a way to:

  1. Record real API interactions
  2. Replay them in tests
  3. Keep my sanity intact

Enter snapshot testing – the closest thing we have to a “time-turner” for API testing.

time-turner

Into the snapshot

Process I have came up with was as follow:

  1. Compatibility testing: Find a good test scenario
  2. Recording mode: Activate special mode in your service
  3. Record snapshots: Capture real API interactions and save them
  4. Test run with snapshot data: Replay recordings in tests using WireMock
  5. Compare results: Validate outputs

Let’s break down the magic:

Compatibility Testing 🔮

(We’ll cover this in detail next time. For now, imagine we’ve stolen a good test case from production like proper coders do.)

Recording mode 🎥

In this part we need to somehow pass information to our service about executing context. We want to tell our service: “Hey, do your normal thing, but also write down everything you see!”.
Two ways (propositions) to do this:

A. solution build configuration Approach

We could create new configuration Snapshot, and use preprocessor as follow:

//some code
#if SNAPSHOT
//code that will be executed only on Snapshot build configuration
#endif

Pros: It’s very easy to be enabled in Rider/VS and rather hard to misconfigure service when not needed snapshot-setting

Cons: More of a niche, and might make Jon Skeet angry.


B. Just configuration variable Approach:

if (Configuration.GetValue<bool>("IsSnapshotMode"))
{
    // Special mode activated!
}

Pros: configuration value in .net apps that are using some kind of IHost builder may be pretty much anything, environment variable, setting file value, command line argument, etc. (Asp.Net core configuration)

Cons: Requires actual adulting with configuration

Record snapshots 📼

So what’s our strategy is, we want to intercept each http client call and save information about it.

As interceptor we will implement custom delegating handler:

public class RecorderDelegatingHandler : DelegatingHandler
{
    protected override async Task<HttpResponseMessage> SendAsync(HttpRequestMessage request, CancellationToken cancellationToken)
    {
        var response = await base.SendAsync(request, cancellationToken);

        // Capture request/response details 
        var address = Uri.UnescapeDataString(request.RequestUri!.PathAndQuery);
        var method = request.Method.ToString();
        var responseStatusCode = (int)response.StatusCode;
        var requestContent = request.Content is null ? string.Empty : await request.Content.ReadAsStringAsync(cancellationToken);
        var responseContent = await response.Content.ReadAsStringAsync(cancellationToken);
        
        return response;
    }
}

Key Notes:

So we do have data describing http call, but still, we need to save it somehow. Let’s create something that will be responsible for that:

public record Snapshot(string Address, string Method, string RequestBody, string ResponseBody, int StatusCode);

public class Recorder : IAsyncDisposable
{
    private readonly ConcurrentBag<Snapshot> _snapshots = [];

    public void AddSnapshot(Snapshot snapshot) => _snapshots.Add(snapshot);

    public async ValueTask DisposeAsync()
    {
        var json = JsonSerializer.Serialize(_snapshots.OrderBy(s => s.Address));
        await File.WriteAllTextAsync("snapshot.json", json);
    }
}

pensieve

Having that we can simply use IHttpContextAccessor in our delegate and create instance of our Recorder class and store snapshots.

What’s left is registration code:

public static IServiceCollection AddSnapshot(this IServiceCollection services){
#if SNAPSHOT
  services.TryAddTransient<RecorderDelegatingHandler>();
  services.AddHttpContextAccessor();
  services.TryAddScoped<Recorder>();

  services.ConfigureAll<HttpClientFactoryOptions>(o => 
  o.HttpMessageHandlerBuilderActions.Add(b => b.AdditionalHandlers.Insert(b.AdditionalHandlers.Count, b.Services.GetRequiredService<RecorderDelegatingHandler>)));
#endif
  return services;
}

Beside registering our delegate and recorder to DI container, we are also configuring all HttpClientFactoryOptions that are present in application. This way, each HttpClient that is going to be used via IHttpClientFactory (so typed clients as well) are going to be supported by our add-on.

Let’s test it, and make two calls in some endpoint:

app.MapGet("/spellsAndIngredients", async (IHttpClientFactory httpClientFactory) =>
{
    var httpClient = httpClientFactory.CreateClient();
    httpClient.BaseAddress = new Uri("https://wizard-world-api.herokuapp.com/");

    var spells = await httpClient.GetFromJsonAsync<Spell[]>("Spells?Type=Spell");
    var ingredients = await httpClient.GetFromJsonAsync<Ingredient[]>("Ingredients?Name=Dragon");
    
    return new {spells, ingredients};
});

After it finishes we can examine our snapshot file:

[
  {
    "Address": "/Ingredients?Name=Dragon",
    "Method": "GET",
    "RequestBody": "",
    "ResponseBody": "[/*...*/]",
    "StatusCode": 200
  },
  {
    "Address": "/Spells?Type=Spell",
    "Method": "GET",
    "RequestBody": "",
    "ResponseBody": "[/*...*/]",
    "StatusCode": 200
  }
]

and as we can see, all calls has been successfully saved into the file.

Test run with snapshot data 🧪

Now for the fun part – replaying our recordings using WireMock.NET. For testing we are going to use WebApplicationFactory.

Our test look like follows:

[Fact]
public async Task Should_Get_Snapshot_Spells()
{
    using var wireMock = WireMockServer.Start(port: 61656, useSSL: false);
    
    // Load our snapshot
    var snapshots = LoadSnapshots("snapshot.json");

    foreach (var snapshot in snapshots)
    {
        wireMock
            .Given(Request.Create()
                .WithRelativePath(snapshot!.Address)
                .WithRequestBody(snapshot.RequestBody)
                .WithMethod(snapshot.Method))
            .RespondWith(Response.Create()
                .WithStatusCode(snapshot.StatusCode)
                .WithBody(snapshot.ResponseBody));
    }

    var httpResponse = await _client.GetAsync("spellsAndIngredients");
    var contentResponse = await httpResponse.Content.ReadAsStringAsync();
    
    await Verify(contentResponse.ToIndentedJson()); // Using Verify library
}

We are starting our in process wire mock instance and create mocks based on our snapshot file.

After mappings/mocks are done, we can simply call our service endpoint - and just Verify lib to test response.

Conclusions 🏁

After implementing this snapshot sorcery, here’s what we’ve learned:

Next Time: We’ll dive into compatibility testing – how to find perfect test scenarios without disturbing real users. Until then, may your tests be green and your mocks be ever-accurate!

Full code example can be found in this repo.