Customize message pumps
While the message processing is handled by the IMessageHandler<>
implementations, the message pump controls in what format the message is received.
We allow several customizations while implementing your own message pump.
Control custom deserialization
When inheriting from an ...MessagePump
type, there's a way to control how the incoming raw message is being deserialized.
Based on the message type of the registered message handlers, the pump determines if the incoming message can be deserialized to that type.
using System;
using Arcus.Messaging.Pumps.Abstractions;
using Microsoft.Extensions.Configuration;
using Microsoft.Extensions.DependencyInjection;
using Microsoft.Extensions.Logging;
public class OrderMessagePump : MessagePump
{
public OrderMessagePump(
IConfiguration configuration,
IServiceProvider serviceProvider,
ILogger<OrderMessagePump> logger)
: base(configuration, serviceProvider, logger)
{
}
public override bool TryDeserializeToMessageFormat(string message, Type messageType, out object? result)
{
if (messageType == typeof(Order))
{
result = JsonConvert.DeserializeObject<Order>(message);
return true;
}
else
{
result = null;
return false;
}
}
}
Filter messages based on message context
When registering a new message handler, one can opt-in to add a filter on the message context which filters out messages that are not needed to be processed.
This can be useful when you are sending different message types on the same queue. Another use-case is being able to handle different versions of the same message type which have different contracts because you are migrating your application.
Following example shows how a message handler should only process a certain message when a property's in the context is present.
We'll use a simple message handler implementation:
using Arcus.Messaging.Abstractions;
using Arcus.Messaging.Pumps.Abstractions.MessageHandling;
public class OrderMessageHandler : IMessageHandler<Order>
{
public async Task ProcessMessageAsync(Order order, MessageContext context, ...)
{
// Do some processing...
}
}
We would like that this handler only processed the message when the context contains MessageType
equals Order
.
using Microsoft.Extensions.DependencyInjection;
public class Startup
{
public void ConfigureServices(IServiceCollection services)
{
services.WithMessageHandler<OrderMessageHandler, Order>(context => context.Properties["MessageType"].ToString() == "Order");
}
}
Note that the order in which the message handlers are registered is important in the message processing. In the example, when a message handler above this one is registered that could also handle the message (same message type) than that handler may be chosen instead of the one with the specific filter.
Fallback message handling
When receiving a message on the message pump and none of the registered IMessageHandler
's can correctly process the message, the message pump normally throws and logs an exception.
It could also happen in a scenario that's to be expected that some received messages will not be processed correctly (or you don't want them to).
In such a scenario, you can choose to register a IFallbackMessageHandler
in the dependency container.
This extra message handler will then process the remaining messages that can't be processed by the normal message handlers.
Following example shows how such a message handler can be implemented:
using Arcus.Messaging.Abstractions;
using Arcus.Messaging.Pumps.Abstractions.MessageHandling;
using Microsoft.Extensions.Logging;
public class WarnsUserFallbackMessageHandler : IFallbackMessageHandler
{
private readonly ILogger _logger;
public WarnsUserFallbackMessageHandler(ILogger<WarnsUserFallbackMessageHandler> logger)
{
_logger = logger;
}
public async Task ProcessMessageAsync(string message, MessageContext context, ...)
{
_logger.LogWarning("These type of messages are expected not to be processed");
}
}
And to register such an implementation:
using Microsoft.Extensions.DependencyInjection;
public class Startup
{
public void ConfigureServices(IServiceCollection services)
{
services.WithFallbackMessageHandler<WarnsUserFallbackMessageHandler>();
}
}