SlimMessageBus - Amazon SQS and SNS Transport
Before diving into this provider documentation, please make sure to read the Introduction.
Table of Contents
- Configuration
- Producing Messages
- Consuming Messages
- Transport-Specific Settings
- Headers
- Request-Response Configuration
- Topology Provisioning
- Future Ideas
Configuration
To configure Amazon SQS / SNS as the transport provider, you need to specify the AWS region and choose an authentication method:
- Ambient Credential: Connect to AWS using default credentials that are pulled from the environment variables (e.g. Fargate, ECS, EC2, etc.).
- Static Credentials: Learn more
- Temporary Credentials: Learn more
using SlimMessageBus.Host.AmazonSQS;
services.AddSlimMessageBus((mbb) =>
{
    mbb.WithProviderAmazonSQS(cfg =>
    {
        cfg.UseRegion(Amazon.RegionEndpoint.EUCentral1);
        // Use static credentials: https://docs.aws.amazon.com/sdkref/latest/guide/access-iam-users.html
        cfg.UseStaticCredentials(accessKey, secretAccessKey, SqsMessageBusMode.All);
        // Use default credentials pulled from environment variables (EC2, ECS, Fargate, etc.):
        // cfg.UseDefaultCredentials(); // This is the default, so you can skip this line if you want to use the default credentials.
        // Use temporary credentials: https://docs.aws.amazon.com/IAM/latest/UserGuide/id_credentials_temp_use-resources.html#RequestWithSTS
        // cfg.UseTemporaryCredentials(roleArn, roleSessionName);
        AdditionalSqsSetup(cfg);
    });
});
For an example configuration, check out this file: SqsMessageBusSettings. The settings allow you to customize the SQS and SNS client object and control topology provisioning for advanced scenarios.
The plugin supports both SQS (Simple Queue Service Queues) and SNS (Simple Notification Service).
However, if you want to use just one specify the mode parameter in either .UseStaticCredentials(mode: SqsMessageBusMode.Sqs) or .UseStaticCredentials(mode: SqsMessageBusMode.Sns).
Producing Messages
To produce a TMessage to an Amazon SQS queue or an SNS topic:
// Send TMessage to Amazon SQS queue
mbb.Produce<TMessage>(x => x.UseQueue());
// OR
// Send TMessage to Amazon SNS topic
mbb.Produce<TMessage>(x => x.UseTopic());
This configuration sends TMessage to the specified Amazon SQS queue or SNS topic. You can then produce messages like this:
TMessage msg;
// Send msg to "some-queue"
await bus.Publish(msg, "some-queue");
// OR
// Send msg to "some-topic"
await bus.Publish(msg, "some-topic");
The second parameter (path) specifies the name of the queue or topic to use.
If you have a default queue or topic configured for a message type:
mbb.Produce<TMessage>(x => x.DefaultQueue("some-queue"));
// OR
mbb.Produce<TMessage>(x => x.DefaultTopic("some-topic"));
You can simply call bus.Publish(msg) without providing the second parameter:
// Send msg to the default queue "some-queue" (or "some-topic")
bus.Publish(msg);
Note that if no explicit configuration is provided, the system assumes the message will be sent to a topic (equivalent to using UseTopic()).
Consuming Messages
To consume messages of type TMessage by TConsumer from an Amazon SQS queue named some-queue:
mbb.Consume<TMessage>(x => x
   .Queue("some-queue")
   //.WithConsumer<TConsumer>());
To consume messages from an Amazon SQS queue some-queue that is subscribed to a some-topic Amazon SNS topic:
mbb.Consume<TMessage>(x => x
   .Queue("some-queue")
   .SubscribeToTopic("some-topic")
   //.WithConsumer<TConsumer>());
The
.SubscribeToTopic()will create a subscription on the SNS and setup a policy on the SQS queue to accept messages from that topic.
Consumer Context
The consumer can get the IConsumerContext interface injected to access native Amazon SQS messages:
public class PingConsumer(IConsumerContext context) : IConsumer<PingMessage>
{
   public Task OnHandle(PingMessage message, CancellationToken cancellationToken)
   {
      // Access the native Amazon SQS message:
      var transportMessage = context.GetTransportMessage(); // Amazon.SQS.Model.Message type
   }
}
This can be helpful to extract properties like MessageId or Attributes from the native SQS message.
Transport-Specific Settings
Producer and consumer configurations have additional settings like:
- EnableFifo
- MaxMessageCount
For a producer:
mbb.Produce<TMessage>(x => x
   .EnableFifo(f => f
     .DeduplicationId((m, h) => (m.Counter + 1000).ToString())
     .GroupId((m, h) => m.Counter % 2 == 0 ? "even" : "odd")
   )
);
For a consumer:
mbb.Consume<TMessage>(x => x
   .WithConsumer<TConsumer>()
   .Queue("some-queue")
   .EnableFifo()
   .MaxMessageCount(10)
   .Instances(1));
These default values can also be set at the message bus level using SqsMessageBusSettings:
mbb.WithProviderAmazonSQS(cfg =>
{
   cfg.MaxMessageCount = 10;
});
Settings at the consumer level take priority over the global defaults.
Headers
Amazon SQS has specific requirements for message headers, detailed in this guide.
Headers in SlimMessageBus (SMB) can be any type (object). To convert them, SMB uses a header serializer, like the default implementation: DefaultSqsHeaderSerializer.
By default, string values are attempted to be converted to Guid, bool, and DateTime.
You can override the serializer through SqsMessageBusSettings.
Request-Response Configuration
Producing Request Messages
When sending a request, you need to specify whether it should be delivered to a topic or queue (see Producing Messages).
To receive responses on a queue:
mbb.ExpectRequestResponses(x =>
{
    x.ReplyToQueue("test-echo-queue-resp");
    x.DefaultTimeout(TimeSpan.FromSeconds(60));
});
Or to receive responses on a topic:
mbb.ExpectRequestResponses(x =>
{
    x.ReplyToTopic("test-echo-resp");
    x.DefaultTimeout(TimeSpan.FromSeconds(60));
});
Each service instance should have a dedicated queue or topic to ensure the response arrives back at the correct instance. This ensures the internal task Task<TResponse> of bus.Send(TRequest) resumes correctly.
Handling Request Messages
For services handling requests, you must configure the request consumption settings for a specific queue (or topic):
Example for queue:
mbb.Handle<EchoRequest, EchoResponse>(x => x
   .Queue(queue)
   .WithHandler<EchoRequestHandler>());
Example for topic:
mbb.Handle<EchoRequest, EchoResponse>(x => x
   .Topic(topic)
   .SubscriptionName("handler")
   .WithHandler<EchoRequestHandler>());
Ensure that if a request is sent to a queue, the consumer is also listening on that queue (and similarly for topics). Mixing queues and topics for requests and responses is not supported.
Topology Provisioning
Amazon SQS can automatically create any queues declared in your SMB configuration when needed. This process occurs when the SMB instance starts, and only for queues that do not yet exist. If a queue already exists, it will not be modified.
Note: Automatic topology creation is enabled by default.
To disable automatic topology provisioning:
mbb.WithProviderAmazonSQS(cfg =>
{
   cfg.TopologyProvisioning.Enabled = false;
});
You can also customize how queues are created by modifying the CreateQueueOptions:
mbb.WithProviderAmazonSQS(cfg =>
{
   cfg.TopologyProvisioning = new()
   {
      CreateQueueOptions = (options) =>
      {
         // Customize queue options here
      }
   };
});
You can control which services create queues:
mbb.WithProviderAmazonSQS(cfg =>
{
   cfg.TopologyProvisioning = new()
   {
      Enabled = true,
      CanProducerCreateQueue = true,  // Only producers can create queues
      CanConsumerCreateQueue = false, // Consumers cannot create queues
   };
});
By default, both flags are enabled (
true).
This flexibility allows you to define ownership of queues/topics clearly—e.g., producers handle queue creation while consumers manage subscriptions.
Future Ideas
Application-to-Person (A2P) Support
The SNS supports sending SMS messages and Email to persons. While this is not a typical system to system communication, there could be few configuration and mapping features that could be added to leverage that capability from SMB. If you need that, please raise an issue (feature request).