Overview

When an exception occurs within a pipeline an OnPipelineException event is raised on the pipeline and any observers that have hooked onto the event will be called:

public class ReceiveExceptionObserver : IPipelineObserver<OnPipelineException>
{
	public void Execute(OnPipelineException pipelineEvent)
	{
		if (pipelineEvent.Pipeline.ExceptionHandled) // <-- set by calling MarkExceptionHandled
		{
			return;
		}

		pipelineEvent.Pipeline.MarkExceptionHandled(); // <-- sets ExceptionHandled to true
		pipelineEvent.Pipeline.Abort(); // <-- prevents further processing of the pipeline
	}
}

Typically you would not respond to the OnPipelineException event but you need to be cognizant of how it affects the message processing.

The more interesting bit is that the first order of business is to determine whether or not to retry the message. If the exception is of type UnrecoverableHandlerException the message will not be retried and will be immediately moved to the ErrorQueue. This means that when you are able to determine that the message will never be processed correctly, but you do not want to discard it outright, you can throw this exception and the message will be moved directly to the ErrorQueue.

If the exception is not of type UnrecoverableHandlerException the implementation of the IServiceBusPolicy on the ServiceBusConfiguration is used to invoke the relevant method. The DefaultServiceBusPolicy makes use of the serviceBus settings in the application configuration file to determine the ‘maximumFailureCount’. Should the number of failed messages still be within this count the message will be retried and duration to wait until the next retry is determined by using the durationToIgnoreOnFailure value.

Should the message be retried the exception message is added to the FailureMessages collection of the TransportMessage and the transport message is re-enqueued, moving it to the end of the queue and the IgnoreTillDate property of the TransportMessage is also set to the duration to wait before retrying.

Note: this is where you may experience some serious queue thrashing if you do not have a DeferredQueue configured for your inbox. When the inbox processor dequeues the message it check to see whether it can be processed by checking the IgnoreTillDate value. If it cannot be processed it is moved to the DeferredQueue if there is one; else it is simply re-enqueued in the inbox. Since the inbox is responsible for processing messages as quickly as possible this processing will continue until the message becomes available for processing. Given that there is a message to process in the inbox no backing-off will occur. Please always use a deferred queue for anything that will be executed in a production environment.