title | reviewed | component | redirects | related | ||||
---|---|---|---|---|---|---|---|---|
Sending messages |
2023-06-02 |
Core |
|
|
NServiceBus supports sending different types of messages (see Messages, Events, and Commands) to any endpoint. Messages can be sent either directly from the endpoint or as part of handling an incoming message. When a message arrives at an endpoint, it goes through a pipeline of processing steps.
In some cases, messages that need to be sent may not be related to an incoming message. Some examples are:
- Sending a command when an HTML form is submitted in an ASP.NET application.
- Publishing an event when the user clicks a button on a GUI application (see Publish and Handle an Event).
To send a message directly from the endpoint:
snippet: BasicSend
Unit testing the process of sending a message is supported by the NServiceBus.Testing
library.
Messages are often sent as part of handling an incoming message. When running in a transaction mode that supports it, these send operations take part in the same transaction as that of the message handler, thereby ensuring that the send operation rolls back if the handling of the message fails at any stage of the message processing pipeline.
To send a message from inside a message handler:
snippet: SendFromHandler
Warning
Using IMessageSession
or IEndpointInstance
to send messages inside a handler instead of the provided IMessageHandlerContext
should be avoided.
Some of the dangers when using an IMessageSession
or IEndpointInstance
inside a message handler to send or publish messages are:
- Those messages will not participate in the same transaction as that of the message handler. This could result in messages being dispatched or events published even if the message handler resulted in an exception and the operation was rolled back.
- Those messages will not be part of the batching operation.
- Those messages will not contain any important message header information that is available via the
IMessageHandlerContext
interface parameter, e.g., CorrelationId.
Note
Send
is an asynchronous operation. When the invocation ends, it does not mean that the message has actually been sent. In scenarios where a large number of messages are sent in a short period, it might be beneficial, from a performance perspective, to limit the number of outstanding send operations pending for completion. Sample approaches that can be used to limit the number of send tasks can be found in Writing Async Handlers.
The SendOptions
object can be used to override the default routing.
Using the destination address:
snippet: BasicSendSetDestination
Using the ID of the target instance:
snippet: BasicSendSpecificInstance
Sending a message to the same endpoint, i.e. sending to self, can be done in two ways.
An endpoint can send a message to any of its own instances:
snippet: BasicSendToAnyInstance
Or, it can request a message to be routed to itself, i.e. the same instance.
Note
This option is only possible when an endpoint instance ID has been specified.
Messages are sent via the queueing infrastructure just like a regular Send. This means that it will use batched dispatch and - if configured - outbox.
snippet: BasicSendToThisInstance
When a receiving endpoint replies to a message, the reply message will be routed to any instance of the sending endpoint by default. The sender of the message can also control how reply messages are received.
To send the reply message to the specific instance that sent the initial message:
snippet: BasicSendReplyToThisInstance
To send the reply message to any instance of the endpoint:
snippet: BasicSendReplyToAnyInstance
The sender can also request the reply to be routed to a specific transport address
snippet: BasicSendReplyToDestination
While it's usually best to let NServiceBus handle all exceptions, there are some scenarios where messages might need to be sent regardless of whether the message handler succeeds or not, for example, to send a reply notifying that there was a problem with processing the message.
This can be done by using the immediate dispatch API:
snippet: RequestImmediateDispatch
Note
The API behaves the same for ITransactionalSession
but differently for IMessageSession
. When invoking message operations on IMessageSession
the RequestImmediateDispatch
does not have any effect as messages will always be immediately dispatched.
Side effects can occur when failures happen after sending the message. The messages could be retried meaning duplicate messages are created if this code is executed more than once.
When messages are sent via immediate disaptch:
- Ensure that the same message identifier gets assigned to them when invoked more than once.
- Due to failures it could happen that messages are sent that contain state which is inconsistent because of failing operations like a storage modification that didn't occur.
By specifying immediate dispatch, outgoing messages will not be batched or enlisted in the current receive transaction, even if the transport supports transactions or batching. Similarly, when the outbox feature is enabled, messages sent using immediate dispatch won't go through the outbox.