I'm struggling to understand why so many examples on the web are using MediatR when explaining CQRS patterns, when dealing with commands and queries.
Almost everywhere I see examples where Commands and Queries are handled by MediatR but I do not see any benefit from it other than not needing to register every Command or Query in Dependency Injection Container. But then you need to implement Query objects (inheriting IRequest), Query Handlers, and Query Responses objects so that in your API controller method you can then call _mediatr.Send(queryObject)
.
Why not just use dependency injection to inject query object into API controller which you can call "get" methods directly on? Like:
[HttpGet]
[Route("getall")]
public async Task<IncidentQueryResult> GetAll(int page = 0, int pageSize = 25)
{
var result = await _incidentQueries.GetIncidents(page, pageSize);
return result;
}
instead of:
[HttpGet]
[Route("getall")]
public async Task<IncidentQueryResult> GetAll(int page = 0, int pageSize = 25)
{
var query = new IncidentQuery(page, pageSize);
var result = await _mediatr.Send(query);
return result;
}
Then, inside the GetIncidents
method, there's a direct sql call to database and mapping results to C# objects. Plain and simple.
For me, the perfect and only reasonable use of MediatR library is to handle Domain Events. When implementing DDD, I'm trying to set up a project in the way presented below. Each rectangle is a different project in the solution. The arrows represents references:
Let's imagine a scenario: Creating a domain object needs to increment a counter stored in another domain object (a different aggregate)
So the MediatR for me works only in Infrastructure and Application layer.
Are people just using MediatR for Commands and Queries just for sake of using it? For me it looks like adding commands and queries handlers, query and command request and response types only adds more code that has no real value and only makes it less understandable.
Here are some links I visited:
A lot of sites in my native language has this as well.
Having too many handlers used throughout your application makes difficult to read what your application does and what triggers what. What I saw was that people were handling domain events in Commands layer but the domain probably shouldn't send commands directly?
Do you need to use MediatR at all in CQRS?
MediatR
is just one library to solve a specific need. As the repository puts it:
In-process messaging with no dependencies.
Command Query Responsibility Segregation (or CQRS
) is a design pattern that can be implemented in many ways. Fundamentally as long as your reads and writes are independent you are "following" this pattern.
So no, you can build an entire app that is CQRS
"compliant" without using MediatR
or any other messaging library. That said you can also build it as one massive file. CQRS
is simply one of many tools for managing code you can deploy depending on your needs.
You can also use MediatR
for in process messeging only and not apply the CQRS
pattern.
That said a common reason you likely see tutorials, blogs and other .NET CQRS
resources use MediatR
(or a similar messaging library) is that generally any app using CQRS
is also going to want the following:
MediatR
generally solves both these problems very nicely by separating out the execution of the command to it's implementation (which can live in a separate project)
In your case for example, Presentation
has to have a knowledge of the database implementation as well as it's schema in order to execute queries as well as mapping them to database resources rather then leaving that as a concern of infrastructure and infrastructure alone. In my experience this can lead to either lots of duplicate code or lots of unintended coupling between projects. Better to have the presentation layer souly focus on presentation and send a message out to whichever service (it does not care which or deal with registering them in MediatR
) can give it Query information upon request.
Essentially in your diagram MediatR
(and/or NServiceBus
, Brighter
, MassTransit
, Rebus
... if you need to scale beyond a single process) would act as a way to control the flow of data and detatch the client of the Query/Command from it's handler.
So finally to answer:
Are people just using MediatR for Commands and Queries just for sake of using it?
Yes and no, they are mostly using it as a separate good practice that plays very nicely together with the CQRS
pattern to control their dependency flow. Although you are right that for many beginners to these ideas they can tie the two together in ways that are not required or recommended.
I recommend you take a look at his other work on Clean Code
to understand how all these pieces play together to create (what he calls) a "pit of success" for future developers working on the project. He has a template repo here and several talks online about it: https://github.com/jasontaylordev/CleanArchitecture