Getting Started
Architecture
Transports
Persistence
ServiceInsight
ServicePulse
ServiceControl
Monitoring
Samples

Messages, events, and commands

Component: NServiceBus

A message is the unit of communication for NServiceBus. There are two types of messages, commands and events, that capture more of the intent and users to follow messaging best-practices.

Commands

A command tells a service to do something, and typically a command should only be consumed by a single consumer. Commands are sent via either message handler context within a message handler, a saga, a pipeline behavior, a message or transactional session. For example if there is a command, such as SubmitOrder, then there should only be one handler or saga that implements IHandleMessages<SubmitOrder>.

Commands should be expressed in a verb-noun sequence, following the tell style:

  • UpdateCustomerAddress
  • UpgradeCustomerAccount
  • SubmitOrder

Events

An event signifies that something has happened. Events are published via either message handler context within a message handler, a saga, a pipeline behavior, a message or transactional session.

Events should be expressed in a noun-verb (past tense) sequence, indicating that something happened. Some example event names may include:

  • CustomerAddressUpdated
  • CustomerAccountUpgraded
  • OrderSubmitted, OrderAccepted, OrderRejected, OrderShipped

Commands vs Events

CommandEvent
Used to make a request to perform an action.Used to communicate that an action has been performed.
Has one logical owner.Has one logical owner.
Should be sent to the logical owner.Should be published by the logical owner.
Cannot be published.Cannot be sent.
Cannot be subscribed to or unsubscribed from.Can be subscribed to and unsubscribed from.
Can be sent using the gateway.Cannot be sent using the gateway.

Validation

There are checks in place to ensure best practices are followed. Violations of the above guidelines generate the following exceptions:

  • "Pub/Sub is not supported for Commands. They should be sent directly to their logical owner." — thrown when attempting to publish a Command or subscribe to/unsubscribe from a Command.
  • "Events can have multiple recipients so they should be published." — thrown when attempting to use Send() to send an event.
  • "Reply is not supported for commands or events. Commands should be sent to their logical owner. Events should be published." — thrown when attempting to reply with a Command or an Event.
  • "Cannot configure routing for type because it is not considered a message. Message types have to either implement NServiceBus.IMessage interface or match a defined message convention." — thrown when configuring the destination endpoint for a non-message type.
  • "Cannot configure routing for assembly because it contains no types considered as messages. Message types have to either implement NServiceBus.IMessage interface or match a defined message convention." — thrown when configuring the destination endpoint for an assembly that contains no types considered messages.
  • "Cannot configure routing for namespace because it contains no types considered as messages..." — thrown when configuring the destination endpoint for a namespace that contains no types considered messages.
  • "Cannot configure publisher for type because it is not considered a message. Message types have to either implement NServiceBus.IMessage interface or match a defined message convention." — thrown when configuring the publisher for a type that is not a message.
  • "Cannot configure publisher for type because it is not considered an event. Event types have to either implement NServiceBus.IEvent interface or match a defined event convention." — thrown when configuring the publisher for a type that is not an event.
  • "Cannot configure publisher for type because it is a command." — thrown when configuring the publisher for a command.

This enforcement is enabled by default but can be disabled.

Designing messages

A message can be defined using a class, record, or an interface. Messages should focus on data only and avoid including methods or other business logic. Treating messages as simple contracts makes them easier to version and evolve over time.

Ideally, a good message type will:

  • Be as small as possible
  • Satisfy the Single Responsibility Principle
  • Favor simplicity and redundancy over object-oriented practices like inheritance
  • Not be re-used for other purposes (e.g., domain objects, data access objects, or UI binding objects)

Generic message definitions (e.g., MyMessage<T>) are not supported. It is recommended to use dedicated, simple types for each message.

Messages define the data contracts between endpoints. More details are available in the sharing message contracts documentation.

By following these guidelines, message types are generally more compatible with serializers and tend to be more evolvable over time.

Identifying messages

Endpoints will process any message that can be deserialized into a .NET type but requires message contracts to be identified upfront to support:

Messages can be defined by implementing a marker interface or specifying a custom convention.

Marker interfaces

The simplest way to identify messages is to use interfaces.

  • NServiceBus.ICommand for a command.
  • NServiceBus.IEvent for an event.
  • NServiceBus.IMessage for any other message type (e.g., a reply in a request/response pattern).
public class MyCommand : ICommand { }

public class MyEvent : IEvent { }

public class MyMessage : IMessage { }

Those interfaces are available in NServiceBus.MessageInterfaces. The project targets netstandard2.0 has a stable version number which is highly unlikely to change. Using these well-defined interfaces should be prefered over conventions since NServiceBus.MessageInterfaces package can be used to create a shared message assembly that can be used by multiple major versions of NServiceBus, and in projects using different target frameworks, while still relying on the ICommand and IEvent marker interfaces.

Conventions

Custom conventions can be used to identify the types used as contracts for messages, commands, and events. This is known as unobtrusive mode.

Samples

  • Sharing message assemblies
    Demonstrate how the NServiceBus.MessageInterfaces package enables message assemblies to be shared across major versions of NServiceBus.

Related Articles

  • Conventions
    Custom conventions for defining how certain things are detected and to support unobtrusive mode.
  • Unobtrusive Mode Messages
    How to avoid referencing NServiceBus assemblies from message assemblies.