A small .Net messaging library built on top of MSMQ for application integration scenarios. Support for transactions, automatic retries, json serialization, and more. Provides a simple and reliable way of integrating applications and services via message queues minus the complexity of a full-on ESB.
- NOTE - MiniBus now appears as mini.bus in nuget package manager
MiniBus offers the following features:
- Send to one or more queues (local and/or remote)
- Read messages synchronously or asynchronously
- Choice of XML or JSON serialization
- Automatic Message Distribution for load balancing
- Enlist in ambient transactions
- Configurable, robust automatic retries
- Move to error queue on failure
- Automatically create local queues
- Simple logging support
- Return error messages back to the read queue
- Copy a message to one or more queues
- View a message body to help debug contents
- Delete a message by its id
- Fail fast option
- Discard failures
11/04/2020 - Nuget package no longer maintained or up-to-date. To get the latest, clone this repository and build it in Visual Studio.
19/02/2018 - The core of MiniBus has now been rewritten to both more maintainable, testable, and reliable.
A bus instance is created via the BusBuilder class using the many options available to configure it to your needs but first create a type to use as a message:
public class PlaceOrder
{
public string OrderNumber { get; set; }
public string Description { get; set; }
}
// create a bus for sending messages
IBus bus = new BusBuilder()
.WithLogging(new FileLogger())
.DefineErrorQueue("MiniBus.errors")
.DefineWriteQueue("MiniBus.messages1")
.DefineWriteQueue("MiniBus.messages2@remotepc")
.DefineWriteQueue("MiniBus.messages3")
.CreateLocalQueuesAutomatically()
.EnlistInAmbientTransactions()
.AutoDistributeOnSend()
.JsonSerialization()
.TimeToBeReceived(TimeSpan.FromMinutes(1))
.CreateBus();
// create your message type
var order = new PlaceOrder { o.OrderNumber = "N0232344", Description = "Fries and a Shake" };
// send it
bus.Send(order);
Alternatively, the Send method now allows via an override the ability to specify which of the previously configured queues to send to.
bus.Send(order, "MiniBus.messages2@remotepc");
In this scenario the message will be sent once to the specified queue only and options such as AutoDistributeOnSend are ignored. This option gives the flexibility to send "Event" style messages when something interesting happens in your application. All that is required is that all possible destination queues are specified at the time the Bus is built.
Create a message handler by implementing the interface IHandleMessage<T>:
class OrderHandler : IHandleMessage<PlaceOrder>
{
public void Handle(PlaceOrder o)
{
// process the message
}
}
Then register it with the bus and call the Receive method for synchronous processing:
// create a bus for receiving messages
int slide = 4000; // time in milliseconds between retries multiplied by the current retry
IBus bus = new BusBuilder()
.WithLogging(new FileLogger())
.DefineErrorQueue("MiniBus.errors")
.DefineReadQueue("MiniBus.messages1")
.JsonSerialization()
.NumberOfRetries(3, slide)
.CreateBus();
// register one or more message handlers
bus.RegisterHandler<PlaceOrder>(new OrderHandler());
// process messages on the read queue synchronously
bus.Receive<PlaceOrder>();
Or ReceiveAsync for asynchronous processing:
// register one or more message handlers
bus.RegisterHandler<PlaceOrder>(new OrderHandler());
// process messages on the read queue asynchronously
bus.ReceiveAsync<PlaceOrder>();
The BusBuilder class is where we do all the configuration of our bus. We then call the CreateBus method to get back an instance we can use to begin sending or receiving messages.
Unlike most libraries out there, I didn't want to force Log4Net or any other logging library on to the user because of
a) versioning issues
b) keeping MiniBus small, lightweight, and simple.
MiniBus simply provides an interface called ILogMessages. The consuming application is responsible for providing an implementation of this interface to wrap around the logging library it is already using which in a console application could be as simple as logging to the console window:
// log to the console
class ConsoleLogger : ILogMessages
{
public void Log(string message)
{
Console.WriteLine(message);
}
}
or alternatively a logging library such as Log4Net:
// log with Log4Net
class FileLogger : ILogMessages
{
public void Log(string message)
{
if(log.IsDebugEnabled)
log.Debug(message);
}
static readonly ILog log = LogManager.GetLogger(typeof(Program));
}
When a message cannot be processed, you don't want to lose it. By defining an error queue MiniBus has somewhere to place the failed message until you're ready to try again. In order to do so, you can create the bus and call ReturnAllErrorMessages to have the messages moved back to the read queue.
When creating the bus in the receiving application, you need to tell MiniBus which queue to read the messages from. DefineReadQueue does just that.
In the sending application, you can tell MiniBus which queue(s) to send the message to via the DefineWriteQueue method. Unlike DefineErrorQueue and DefineReadQueue, DefineWriteQueue can be called many times. MiniBus will send a message to each queue that you declare with the DefineWriteQueue method.
An alternative that allows all write queues to be defined in one call.
Queues can be on the same machine as the application or on a remote machine. Remote queues are declared in each of the Define<whatever>Queue methods by using the @ syntax. For example myqueue@remotepc. Local queues do not include the @ symbol or machine name. If you declare local queues and they do not exist, calling the CreateLocalQueuesAutomatically method will ensure the queues are created for you before you use the bus to send or receive messages.
By default, MiniBus serializes your messages as Xml. If you would rather use JSON then call this method on the BusBuilder object. When sending a message using XML serialization you'll need to share the message type between the sending and receiving applications as the type is embedded in the XML body as the root element. However, JSON serialization is much looser. There is no requirement to share the type between both endpoints. Deserialization of a message using JSON only looks for matching members between the data in the message body and the type you specify as the target for the deserialization process. Basically this means you can just write a seperate class in each application that look the same (or similar) for both serialization and deserialization without needing to share a dll containing a type.
Sometimes you only want a message to be sent or received as part of a larger transaction, for example, if you are also inserting/updating a row in a database. This option ensures that the operations are treated atomically. In other words, when sending, the message will only be sent if the database operation succeeds. Conversely, when receiving, the message will only be removed if the database operation succeeds. A failure of the database during a read will cause the message to be moved to the error queue.
MiniBus by default will move a failed message to the error queue as soon as an error is detected. Sometimes, perhaps due to network latency in a web service call, the operation would succeed if tried again. The NumberOfRetries method let's you specify how many times MiniBus should retry the operation before giving up and moving the message to the error queue. Additionally, two optional parameters are avaible. The first, slidingRetryInterval, has been added to allow a period of time to pass before a retry should occur. The second, environmentalErrorsOnly, allows you to specify that retries should occur only when an error is believed to be recoverable and might succeed on the next attempt, for example, when a deadlock or timeout occurs. In general, errors that are caused by incorrect data are never going to succeed no matter how many attempts are made so should probably be allowed to fail instantly.
By default, when sending messages MiniBus will send the same message to all the writeQueues you have defined. Sometimes though you may want to evenly distribute messages bewteen all the queues in order to achieve load balancing with the receivers. Setting this option will cause MiniBus to send a message to a different queue on each successive call to Send.
Ordinarily when an error occurs processing a message it is moved to the designated error queue for later intervention. In most cases this is fine but sometimes it can be critical that message order is preserved. In other words, we don't want any subsequent messages to be processed until the failed message is fixed. The FailFast option causes the queue to immediately stop processing messages in the event of an error. The failing message itself stays on the read queue rather than being moved to the error queue so that when the issue is fixed it will be the first message to be processed.
Sometimes if a message fails you just don't care. Either it wasn't important or you know the same message will show up again soon, perhaps from a process that resends until a given state is determined. In this case setting the DiscardFailedMessages option will not put the failed message on the error queue, instead it will just throw it away leaving the error queue maintenance free.
Provides a hook to allow you to specify code that should be executed (asynchronously) if a message is placed on the error queue. Useful, for instance, for sending emails to notify an administrator.
Left unattended and with no administrator monitoring MSMQ can grind to a halt with an "insufficient resources" error. This occurs when the system32\msmq\storage directoy has grown too large with copies of each and every message placed on the "Journal Messages" system queue. Using the AutoPurgeSystemJournal option tells MiniBus to attempt to clean up this queue every fifteen minutes in order to reduce the chances of the error occuring.
Specifies whether a copy of the message should be stored on the originating PC
Specifies how long a message should exist on the queue for before automatically being removed and disposed
Copy a message specified by its Id to another queue
Delete a message specified by its Id from the Read queue
View the contents of a message specified by its Id as a plain string which can be pasted straight into a JSON or XML editor to assist with debugging
If you want to build the source, clone the repository, and open up MiniBus.sln.
git clone https://github.com/SteveBate/MiniBus.git
explorer path/to/MiniBus/MiniBus.sln
MiniBus targets .NET 4.0 and with regard to MSMQ, Windows 7, 8, Server 2008 and above
Feel free to submit an issue on the repository, or find me at @Steve_Bate