The MSMQ quick start application demonstrates how to use asynchronous messaging to implement a system for purchasing a stock. Is follows the same basic approach as in the NMS QuickStart but is adapted as need for use with MSMQ. Please read the introduction in that chapter to get an overview of the system.
When there is direct overlap in functionality between the MSMQ and NMS quickstart a reference to the appropriate section in the NMS QuickStart documentation is given.
Note | |
---|---|
To follow this MSMQ QuickStart load the solution file found in the
directory
|
To communicate between th client and server a pair of queues will be
used. Messages sent from the client to the server will use the
transactional queue named .\Private$\request.txqueue
.
Messages sent from the server to the client will use the transactional
queue .\Private$\response.joe.txqueue
. The queue for
messages that cannot be processed, so called 'poison messages' will be
sent to the queue .\Private$\dead.txqueue
. You can
create these queues using the computer management administration console.
Private queues are used to simplify the application setup
requirements.
Note | |
---|---|
You must create the queues mentioned previously using standard Windows Computer Management console to manage MSMQ. This article covers the basics of creating the queus in the management console. |
Since MSMQ does not natively support the publish-subscribe messaging style as in other messaging systems, Apache MQ, IBM Websphere MQ, TIBCO EMS, the market data information is sent on the same queue as the responses from the server to the client for trade requests..
The gateway interfaces are the same as those described in the NMS QuickStart here.
TradeRequest and TradeResponse messages are defined using XML Schema and classes are generated from that schema. This is the same approach as described in more details in the NMS QuickStart here.
An important difference in the types of message data formats supported 'out-of-the-box' with Apache, IBM, TIBCO as compared to Microsoft MSMQ is the latter support sending a hashtable data structure. As a result, the hashtable that was used to send market data information from the server to the client was changed to be of type System.String in the MSMQ example.
The message handlers are the same as used in the NMS QuickStart here, aside from the change of the hashtable data structure to a string. This is an important benefit of enforcing a separation between the messaging specific classes and the business processing layer.
The message converter used is Spring.Messaging.Support.Converters.XmlMessageConverter. It is configured by specifying the data types that will be send and received. Here is a configuration example for types generated from the XML Schema and a plain string.
<object id="xmlMessageConverter" type="Spring.Messaging.Support.Converters.XmlMessageConverter, Spring.Messaging"> <property name="TargetTypes"> <list> <value>Spring.MsmqQuickStart.Common.Data.TradeRequest, Spring.MsmqQuickStart.Common</value> <value>Spring.MsmqQuickStart.Common.Data.TradeResponse, Spring.MsmqQuickStart.Common</value> <value>System.String, mscorlib</value> </list> </property> </object>
The implementations of the gateway interfaces inherit from Spring's
helper class MessageQueueGatewaySupport
in order to get
easy access to a MessageQueueTemplate
for sending. The
implementation of the IStockService
interface is shown
below
public class MsmqStockServiceGateway : MessageQueueGatewaySupport, IStockService { private Random random = new Random(); private string defaultResponseQueueObjectName; public string DefaultResponseQueueObjectName { set { defaultResponseQueueObjectName = value; } } public void Send(TradeRequest tradeRequest) { MessageQueueTemplate.ConvertAndSend(tradeRequest, delegate(Message message) { message.ResponseQueue = GetResponseQueue(); message.AppSpecific = random.Next(); return message; }); } private MessageQueue GetResponseQueue() { return MessageQueueFactory.CreateMessageQueue(defaultResponseQueueObjectName); } }
The Send
method is using MessageQueueTemplate's
ConvertAndSend(object obj, MessagePostProcessorDelegate
messagePostProcessorDelegate)
method. The anonymous delegate
allows you to modify the message properties, such as ResponseQueue and
AppSpecific after the message has been converted from an object but before
it has been sent. The use of an anonymous delegate allows makes it very
easy to apply any post processing logic to the converted message.
The configuration for MsmqStockServiceGateway
and
all its dependencies is shown below, highlighting important dependency
links.
<object name="stockServiceGateway" type="Spring.MsmqQuickStart.Client.Gateways.MsmqStockServiceGateway, Spring.MsmqQuickStart.Client"> <property name="MessageQueueTemplate" ref="messageQueueTemplate"/> <property name="DefaultResponseQueueObjectName" value="responseTxQueue"/> </object> <object id="messageQueueTemplate" type="Spring.Messaging.Core.MessageQueueTemplate, Spring.Messaging"> <property name="DefaultMessageQueueObjectName" value="requestTxQueue"/> <property name="MessageConverterObjectName" value="xmlMessageConverter"/> </object> <object id="xmlMessageConverter" type="Spring.Messaging.Support.Converters.XmlMessageConverter, Spring.Messaging"> <property name="TargetTypes"> <list> <value>Spring.MsmqQuickStart.Common.Data.TradeRequest, Spring.MsmqQuickStart.Common</value> <value>Spring.MsmqQuickStart.Common.Data.TradeResponse, Spring.MsmqQuickStart.Common</value> <value>System.String, mscorlib</value> </list> </property> </object> <object id="requestTxQueue" type="Spring.Messaging.Support.MessageQueueFactoryObject, Spring.Messaging"> <property name="Path" value=".\Private$\request.txqueue"/> <property name="MessageReadPropertyFilterSetAll" value="true"/> </object> <object id="responseTxQueue" type="Spring.Messaging.Support.MessageQueueFactoryObject, Spring.Messaging"> <property name="Path" value=".\Private$\response.joe.txqueue"/> <property name="MessageReadPropertyFilterSetAll" value="true"/> </object>
Since the client also needs to listen to incoming messages on the
responseTxQueue, a
TransactionalMessageListenerContainer
is configured.
The configuration for the message listener container and all its
dependencies is shown below, highlighting important dependency
links.
<!-- MSMQ Transaction Manager --> <object id="messageQueueTransactionManager" type="Spring.Messaging.Core.MessageQueueTransactionManager, Spring.Messaging"/> <!-- Message Listener Container that uses MSMQ transactional for receives --> <object id="transactionalMessageListenerContainer" type="Spring.Messaging.Listener.TransactionalMessageListenerContainer, Spring.Messaging"> <property name="MessageQueueObjectName" value="responseTxQueue"/> <property name="PlatformTransactionManager" ref="messageQueueTransactionManager"/> <property name="MessageListener" ref="messageListenerAdapter"/> <property name="MessageTransactionExceptionHandler" ref="sendToQueueExceptionHandler"/> </object> <!-- Delegate to plain CLR object for message handling --> <object id="messageListenerAdapter" type="Spring.Messaging.Listener.MessageListenerAdapter, Spring.Messaging"> <property name="HandlerObject" ref="stockAppHandler"/> <property name="DefaultHandlerMethod" value="Handle"/> <property name="MessageConverterObjectName" value="xmlMessageConverter"/> </object> <object id="sendToQueueExceptionHandler" type="Spring.Messaging.Listener.SendToQueueExceptionHandler, Spring.Messaging"> <property name="MessageQueueObjectName" value="deadTxQueue"/> </object> <object id="deadTxQueue" type="Spring.Messaging.Support.MessageQueueFactoryObject, Spring.Messaging"> <property name="Path" value=".\Private$\dead.queue"/> <property name="MessageReadPropertyFilterSetAll" value="true"/> </object>
A similar configuration is used on the server to configure the class
Spring.MsmqQuickStart.Server.Gateways.MarketDataServiceGateway
that implements the IMarketDataService
interface and a TransactionalMessageListenerContainer
to process messages on the requestTxQueue. You can increase the number of
processing thread in the
TransactionalMessageListenerContainer
by setting the
property MaxConcurrentListeners
, the default value is
1.
To run both the client and server make sure that you select 'Multiple Startup Projects' within VS.NET. The GUI has a button to make a hard coded trade request and show confirmation in a text box. A text area is used to display the market data. There is a 'Get Portfolio' button that is not implemented at the moment. A picture of the GUI after it has been running for a while and trade has been sent and responded to is shown below.