DEV Community

Akash Lomas
Akash Lomas

Posted on

Moving Beyond MediatR: Implementing Cross-Cutting Concerns with Native .NET Dependency Injection

For years, the MediatR package has been a default inclusion in almost every new .NET project template. It is frequently hailed as the gold standard for decoupling controllers or minimal APIs from business logic and implementing Clean Architecture or CQRS patterns.

However, as ASP.NET Core has matured, many of the architectural justifications for adding MediatR as a heavy third-party dependency have faded. If you are building standard CRUD or even moderately complex enterprise APIs, it might be time to ask yourself,

Why am I routing my HTTP requests through an abstract in-memory bus when native alternatives exist?

Let’s explore why MediatR might be an unnecessary abstraction in your codebase and how you can replace its most beloved feature Pipeline Behaviors using pure .NET Dependency Injection (DI) plumbing and the Decorator pattern.

The Core Critique of MediatR

While MediatR provides an elegant mechanism for decoupling, it introduces specific friction points into a code-base,

  • Obfuscated Control Flow: Because handlers are resolved dynamically, it is impossible to use standard “Go to Definition” (F12) in your IDE to trace execution directly from an endpoint to its handler. You are forced to search for the corresponding IRequestHandler type implementation.
  • Debugging Overhead: Stepping through code becomes a tedious exercise of bypassing internal library code generator stacks instead of moving sequentially through your business logic.
  • Unnecessary Architecture: In standard Web APIs, the mapping between an endpoint and its handler is almost always 1:1. Introducing a mediator pattern for a direct request-response cycle over complicates the system with minimal structural payback.

If you are using Minimal APIs or standard Controllers, you already have powerful endpoint routing. So why use MediatR? The answer almost always boils down to one critical feature, Pipeline Behaviors.

The “Killer Feature”: Cross-Cutting Concerns

Developers love MediatR because it makes addressing cross-cutting concerns, such as centralized logging, validation pipelines (e.g., using FluentValidation), metrics collection, and OpenTelemetry tracking incredibly clean.

Instead of cluttering every single business handler with repetitive try-catch blocks or explicit validation calls, MediatR lets you define a generic middleware pipeline around your requests:

// The classic MediatR approach
builder.Services.AddMediatR(cfg => {
    cfg.RegisterServicesFromAssembly(typeof(Program).Assembly);
    cfg.AddOpenBehavior(typeof(LoggingBehavior<,>));
    cfg.AddOpenBehavior(typeof(ValidationBehavior<,>));
});
Enter fullscreen mode Exit fullscreen mode

It is a fantastic architectural model, but you do not need a third-party package to achieve it. Modern .NET allows you to implement this exact pattern natively using compile-time type-safe decorators.

The Native Alternative: DI-Based Decoration

Instead of an abstract pipeline behavior model, we can leverage the Decorator Pattern natively supported by the Microsoft.Extensions.DependencyInjection container. This allows us to transparently wrap any standard interface registration with cross-cutting behaviors.

  • Defining a Domain-Driven Request Handler Interface

First, let’s establish our own lightweight, explicit handler contract. This preserves your CQRS separation without binding your domain to external packages.

public interface IRequestHandler<in TRequest, TResponse>
{
    Task<TResponse> HandleAsync(TRequest request, CancellationToken cancellationToken = default);
}
Enter fullscreen mode Exit fullscreen mode
  • Creating a Concrete Business Handler

Your concrete query or command handlers remain pristine. They are entirely unaware of logging, telemetry, or validation logic, adhering perfectly to the Single Responsibility Principle.

public record CreateProductCommand(string Name, decimal Price) : IRequest;

public class CreateProductHandler : IRequestHandler<CreateProductCommand, Guid>
{
    private readonly IProductRepository _repository;

    public CreateProductHandler(IProductRepository repository)
    {
        _repository = repository;
    }

    public async Task<Guid> HandleAsync
      (  CreateProductCommand request, 
         CancellationToken cancellationToken = default)
    {
        var id = Guid.NewGuid();
        // Core business logic goes here...
        return id;
    }
}
Enter fullscreen mode Exit fullscreen mode
  • Implementing the Cross-Cutting Decorator

Now, let’s implement a generic decorator that intercepts the request execution. It implements the same interface but accepts the inner concrete handler as a dependency, wrapping it with infrastructure logic.

public class LoggingHandlerDecorator<TRequest, TResponse> : IRequestHandler<TRequest, TResponse>
{
    private readonly IRequestHandler<TRequest, TResponse> _inner;
    private readonly ILogger<LoggingHandlerDecorator<TRequest, TResponse>> _logger;

    public LoggingHandlerDecorator(
        IRequestHandler<TRequest, TResponse> inner, 
        ILogger<LoggingHandlerDecorator<TRequest, TResponse>> logger)
    {
        _inner = inner;
        _logger = logger;
    }

    public async Task<TResponse> HandleAsync
      (TRequest request, CancellationToken cancellationToken = default)
    {
        var requestName = typeof(TRequest).Name;
        _logger.LogInformation("Executing request: {RequestName}", requestName);

        try
        {
            var response = await _inner.HandleAsync(request, cancellationToken);
            _logger.LogInformation("Successfully executed request: {RequestName}", requestName);
            return response;
        }
        catch (Exception ex)
        {
            _logger.LogError(ex, "Request failed: {RequestName}", requestName);
            throw;
        }
    }
} 
Enter fullscreen mode Exit fullscreen mode

Wiring It Together Natively in Program.cs

To register these decorators without external help, we can write a clean extension method using the native DI container’s service factory capabilities. This gives us full architectural control over which handlers get decorated and in what order.

public static class RequestHandlerRegistrationExtensions
{
    public static IServiceCollection AddDecoratedRequestHandler<TRequest, TResponse, THandler>(
        this IServiceCollection services) 
        where THandler : class, IRequestHandler<TRequest, TResponse>
    {
        // 1. Register the concrete handler with its own concrete type
        services.AddTransient<THandler>();

        // 2. Register the interface using a factory method that constructs the decorator chain
        services.AddTransient<IRequestHandler<TRequest, TResponse>>(sp =>
        {
            var concreteHandler = sp.GetRequiredService<THandler>();
            var logger = sp.GetRequiredService<ILogger<LoggingHandlerDecorator<TRequest, TResponse>>>();

            // Wrap the core handler with our logging decorator
            return new LoggingHandlerDecorator<TRequest, TResponse>(concreteHandler, logger);
        });

        return services;
    }
}
Enter fullscreen mode Exit fullscreen mode

Pro-Tip: If manual registration feels too verbose for thousands of handlers, you can use assembly scanning utilities like Scrutor to dynamically apply the .Decorate() method across your entire service collection automatically.

Your application endpoints remain perfectly clean, directly consuming the generic interface via native dependency injection:

app.MapPost("/products", async (
    CreateProductCommand command, 
    IRequestHandler<CreateProductCommand, Guid> handler) =>
{
    var result = await handler.HandleAsync(command);
    return Results.Ok(result);
});
Enter fullscreen mode Exit fullscreen mode

Architectural Advantages

By shifting from an in-memory mediator library to native DI decoration, you unlock major engineering benefits:

  • No Black Boxes: Your execution pathway is clear. If you place a breakpoint inside your API endpoint and step into handler.HandleAsync(), you will sequentially enter the LoggingDecorator, step through any validation layers, and land cleanly inside your actual business logic handler.
  • Zero Third-Party Vendor Lock-in: Your core business logic blocks do not depend on external packages. Upgrading your framework version will never be delayed due to a breaking change in an open-source messaging mediator.
  • Granular Structural Control: Unlike globally forced middleware pipelines, you can explicitly choose which command or query handlers receive specific decorators. If a high-throughput endpoint requires raw performance without validation overhead, you can register it without its decorator wrap.

Summary

MediatR has served the .NET community exceptionally well over the last decade. However, frameworks evolve. With modern ASP.NET Core DI plumbing, we can maintain clean separation of concerns, enforce CQRS principles, and leverage pipeline behaviors seamlessly using standard patterns built straight into the runtime.

Evaluate your current architecture.

Is MediatR solving a foundational problem for you, or is it merely acting as boilerplate code that your native framework can already handle?

Top comments (0)