This site explains how ServiceMix implements and performs Normalized Message Routing in terms of both JBI- and ServiceMix-defined interfaces and classes.
Before reading this page, please have a look at the explanation of JBI, especially the Normalized Message Router.
See also How Stuff Works.
Principal Steps
Typically, normalized message routing consists of the following steps:
- A component acting as Consumer sends a MessageExchange to its DeliveryChannel (methods:
AsyncBaseLifeCycle.sendConsumerMessageExchange
for asynchronous, several others, component-dependent ones for synchronous delivery) - The
DeliveryChannel
passes theMessageExchange
to the NMR (method:send
,sendSync
) - The NMR chooses a Flow through which to send the
MessageExchange
- The
Flow
selects the Component which is the destined Provider (method:AbstractFlow.doRouting
) - The
Flow
passes theMessageExchange
to theDeliveryChannel
of the component which is/has been found to be the Provider (method:processInBound
. - The
MessageExchange
is either passed via push (if the Component implements {{MessageExchange}}Listener) or being put in a queue, from which the component can pull it.
Principal Classes and Interfaces
ServiceMix provides the following classes and interfaces relating to normalized message routing:
Broker
and its subclasses corresponding to the NMRFlow
is a ServiceMix-interface for different dispatch policiesDeliveryChannelImpl
is theDeliveryChannel
implementationMessageExchangeImpl
is the implementation ofMessageExchange
Broker and Implementations
The core ServiceMix NMR interface is the Broker. It has implementations: the DefaultBroker and the SecuredBroker, which adds basic authorization. For the later, see also Security.
MessageExchange and Implementation
A MessageExchange (ME
) is a JBI-defined interface. It serves as a container for normalized messages following an exchange pattern, which specifies names, sequences and cardinalities for an excange.
Its ServiceMix implementation is MessageExchangeImpl.
Roles of an ME
As pointed out in the JBI page, an ME can be in different roles. A Role.PROVIDER
indicates that an ME is being sent from a consumer to a provider, whereas a Role.CONSUMER
means that the ME goes from the provider to the consumer.
The ServiceMix implementation never changes the role for a given instance of the MessageExchangeImpl
class. Instead, the consumer component and the provider component are given two different objects, which are related to one another via the mirror
property.
This is done to enforce exchange ownership; a component can not modify the exchange if it is not the owner. This is controlled by the CAN_OWNER
flag and checked at several locations, e.g. various setter-methods.
MessageExchange instances are owned by either the initiator, servicer,
or NMR. Creating or accepting a message exchange instance confers
ownership on the creating (or accepting) component. Upon calling send()
for a particular ME instance, the component surrenders ownership of the
ME instance immediately. sendSync() causes ownership to be surrendered
during the course of the call; when the call ends (returning true) the
component resumes ownership of the ME instance.Ownership of an ME instance gives the owner the right to read from and
write to the instance. When a component (or the NMR) doesn't have
ownership of an ME instance, it MUST NOT read from or write to the
instance, except for reading the message exchange status, which can be
queried regardless of ownership. If a component that is not the current
owner of the instance does attempt to call any method (other than
getStatus()) on the instance an appropriate
java.lang.IllegalStateException MUST be thrown.
(Taken from the JSR 208 Specification)
States of an ME
Depending on which kind of pattern was chosen, an ME
can be in different states. In ServiceMix, these states depend on the ME}'s role. A given state encodes an {{ME
's properties and allows or disallows certain operations.
ServiceMix models possible states via a 2-dimensional integer array (int [][]states
). It is only being set in the constructor of MessageExchangeImpl
and differs depending on the type of the ME (PROVIDER
vs. CONSUMER
).
Every row of the array corresponds to a description of a state with all possible succeeding states. The ownership mentioned above is also described within the states. The current state is being kept in an index variable pointing to the current state/row.
When being sent, the methods:
Code Block | ||||
---|---|---|---|---|
| ||||
public void handleSend(boolean sync) throws MessagingException |
...and:
Code Block | ||||
---|---|---|---|---|
| ||||
public void handleAccept() throws MessagingException |
... change the current state depending on the value of the ExchangeStatus and whether there is a Fault.
DeliveryChannel and Implementations
A DeliveryChannel
(DC
) provides for asynchronous, bi-directional communication between a Component
and the NMR.
A DC
is calling handleSend
during its send
and sendSync
methods.
It is calling handleAccept
during push-delivery in processInBound
and accept
.