Shuttle.Core.Mediator
PM> Install-Package Shuttle.Core.Mediator
The Shuttle.Core.Mediator package provides a mediator pattern implementation.
Configuration
In order to get all the relevant bits working you would need to register the IMediator
dependency along with all the relevant IParticipant
dependencies.
You can register the mediator using IServiceCollection
:
services.AddMediator(builder =>
{
builder.AddParticipants(assembly);
builder.AddParticipant<Participant>();
builder.AddParticipant(participantType);
builder.AddParticipant<Message>(participant);
builder.AddParticipant(async (IParticipantContext<T> context) =>
{
await Task.CompletedTask.ConfigureAwait(false);
});
IMediator
The core interface is the IMediator
interface and the default implementation provided is the Mediator
class.
Participants types are instatiated from the IServiceProvider
instance. This means that it depends on how you register the type as to the behaviour.
Task SendAsync(object message, CancellationToken cancellationToken = default);
The SendAsync
method will find all participants that implement the IParticipant<T>
with the type T
of the message type that you are sending.
Participant implementations
public interface IParticipant<in T>
{
Task ProcessMessageAsync(IParticipantContext<T> context);
}
A participant would handle the message that is sent using the mediator. There may be any number of participants that process the message.
Design philosophy
There are no request/response semantics and the design philosophy here is that the message encapsulates the state that is passed along in a pipes & filters approach.
However, you may wish to make use of one of the existing utility classes:-
RequestMessage<TRequest>
The only expectation from a RequestMessage<TRequest>
instance is either a success or failure (along with the failure message).
RequestResponseMessage<TRequest, TResponse>
The RequestResponseMessage<TRequest, TResponse>
takes an initial TRequest
object and after the mediator processing would expect that there be a TResponse
provided using the .WithResponse(TResponse)
method. The same success/failure mechanism used in the RequestMessage<TRequest>
class is also available on this class.
Considerations
If you have a rather predictable sequential workflow and you require something with faster execution then you may wish to consider the Shuttle.Core.Pipelines package.
Performing a benchmark for your use-case would be able to indicate the more suitable option.