Managing Associations
When a saga manages a transaction across multiple domain concepts, such as Order, Shipment, Invoice, etc, that saga needs to be associated with instances of those concepts. An association requires two parameters: the key, which identifies the type of association (Order, Shipment, etc) and a value, which represents the identifier of that concept.
Associating a saga with a concept is done in several ways. First of all, when a Saga is newly created when invoking a @StartSaga
annotated event handler, it is automatically associated with the property identified in the @SagaEventHandler
method. Any other association can be created using the SagaLifecycle.associateWith(String key, String/Number value)
method. Use the SagaLifecycle.removeAssociationWith(String key, String/Number value)
method to remove a specific association.
Note
The API to associate domain concepts within a Saga intentionally only allows a
String
or aNumber
as the identifying value, since aString
representation of the identifier is required for the association value entry which is stored. Using simple identifier values in the API with a straightforwardString
representation is by design, as aString
column entry in the database makes the comparison between database engines simpler. It is thus intentionally that there is noassociateWith(String, Object)
for example, as the result of anObject#toString()
call might provide unwieldy identifiers.
Imagine a saga that has been created for a transaction around an Order. The saga is automatically associated with the Order, as the method is annotated with @StartSaga
. The saga is responsible for creating an Invoice for that Order, and tell Shipping to create a Shipment for it. Once both the Shipment have arrived and the Invoice has been paid, the transaction is completed and the saga is closed.
Here is the code for such a Saga:
By allowing clients to generate an identifier, a saga can be easily associated with a concept, without the need to a request-response type command. We associate the event with these concepts before publishing the command. This way, we are guaranteed to also catch events generated as part of this command. This will end this saga once the invoice is paid and the shipment has arrived.