Upgrade NServiceBus downstreams from Version 7 to 8

Component: NServiceBus

Information on breaking changes affecting maintainers of downstream components like custom/community transports, persistence, and message serializers.

Serializers: Deserialization using ReadOnlyMemory of byte

Deserialization updated to use ReadOnlyMemory<byte> instead of Stream with on Deserialize API method on IMessageSerializer.

object[] Deserialize(ReadOnlyMemory<byte> body, IList<Type> messageTypes = null);

Serializers that do not support ReadOnlyMemory<byte> as deserialization input should avoid .ToArray calls to prevent unnecessary memory allocations. Instead, it's recommended to implement a shim exposing read-only data as types supported by the serializer APIs or use existing implementations like ReadOnlyMemoryExtensions.AsStream from the Microsoft.Toolkit.HighPerformance.

Transports

Transports: Outgoing message TransportOperation API

The outgoing message body passed to the transport via TransportOperation has a new constructor:

public TransportOperation(string messageId, DispatchProperties properties, ReadOnlyMemory<byte> body, Dictionary<string, string> headers)

Transports: Incoming message MessageContext API

A message body passed by the transport to the core using ReadOnlyMemory<byte> instead of byte[]. The MessageContext becomes:

public MessageContext(string nativeMessageId, Dictionary<string, string> headers, ReadOnlyMemory<byte> body, TransportTransaction transportTransaction, ContextBag context)

For transports that use low allocation types, this allows passing message body without additional memory allocations. Secondly, this ensures that the body passed by the transport cannot be changed by code executing in the pipeline (immutable message body).

Persisters

Persister Outbox API: TransportOperation based on ReadOnlyMemory<byte>

The outbox TransportOperation is using ReadOnlyMemory<byte> instead of byte[].


Last modified