Implementing Event Sourced Entities
TODO: add illustrations from the local console
Event Sourced Entities are components that persist their state using the Event Sourcing Model. Instead of persisting the current state, they persist all the events that led to the current state. Akka stores these events in a journal. Event Sourced Entities persist their state with ACID semantics, scale horizontally, and isolate failures.
An Event Sourced Entity must not update its in-memory state directly as a result of a command. The handling of a command, if it results in changes being required to state, should persist events. These events will then be processed by the entity, at which point the in-memory state can and should be changed in response.
When you need to read state in your service, ask yourself what events should I be listening to?. When you need to write state, ask yourself what events should I be persisting?
🎓 To deepen your understanding of event sourcing, consider the free online course Reactive Architecture: CQRS & Event Sourcing in Akkademy. |
TODO: add an illustration
To load an Entity, Akka reads the journal and replays events to compute the Entity’s current state. As an optimization, by default, Event Sourced Entities persist state snapshots periodically. This allows Akka to recreate an Entity from the most recent snapshot plus any events saved after the snapshot.
In contrast with typical create, read, update (CRUD) systems, event sourcing allows the state of the Entity to be reliably replicated to other services. Event Sourced Entities use offset tracking in the journal to record which portions of the system have replicated which events.
Event Sourced Entities persist changes as events and snapshots. Akka needs to serialize that data to send it to the underlying data store. However, we recommend that you do not persist your service’s public API messages. Persisting private API messages may introduce some overhead when converting from a public message to an internal one but it allows the logic of the service public interface to evolve independently of the data storage format, which should be private.
The steps necessary to implement an Event Sourced Entity include:
-
Model the entity’s state and its domain events.
-
Implementing behavior in command and event handlers.
The following sections walk through these steps using a shopping cart service as an example (working sample can be downloaded as a zip file).
Modeling the Entity
Through our "Shopping Cart" Event Sourced Entity we expect to manage our cart, adding and removing items as we please. Being event-sourced means it will represent changes to state as a series of domain events. Let’s have a look at what kind of model we expect to store and the events our entity might generate.
public record ShoppingCart(String cartId, List<LineItem> items, boolean checkedOut) { (1)
public record LineItem(String productId, String name, int quantity) { (2)
public LineItem withQuantity(int quantity) {
return new LineItem(productId, name, quantity);
}
}
}
1 | Our ShoppingCart is fairly simple, being composed only by a cartId and a list of line items. |
2 | A LineItem represents a single product and the quantity we intend to buy. |
Above we are taking advantage of the Java record to reduce the amount of boilerplate code, but you can use regular classes so long as they can be serialized to JSON (e.g. using Jackson annotations).
|
Another fundamental aspect of our entity will be its domain events. For now, we will have 3 different events ItemAdded
, ItemRemoved
and CheckedOut
, defined as below:
public sealed interface ShoppingCartEvent { (1)
@TypeName("item-added") (2)
record ItemAdded(ShoppingCart.LineItem item) implements ShoppingCartEvent {
}
@TypeName("item-removed")
record ItemRemoved(String productId) implements ShoppingCartEvent {
}
@TypeName("checked-out")
record CheckedOut() implements ShoppingCartEvent {
}
}
1 | The 3 types of event all derive from the same type ShoppingCartEvent . |
2 | Includes the logical type name using @TypeName annotation. |
The use of logical names for subtypes is essential for maintainability purposes. Our recommendation is to use logical names (i.e. @TypeName ) that are unique per Akka service. Check type name documentation for more details.
|
Identifying the Entity
In order to interact with an Entity in Akka, we need to assign a component id and an instance id:
-
component id is a unique identifier for all entities of a given type. To define the component id, the entity class must be annotated with
@ComponentId
and have a unique and stable identifier assigned. -
id, on the other hand, is unique per instance. The entity id is used in the component client when calling the entity from for example an Endpoint.
As an example, an entity representing a customer could have the component id customer
and a customer entity for a specific customer could have the UUID instance id 8C59E488-B6A8-4E6D-92F3-760315283B6E
.
The component id and entity id cannot contain the reserved character | , because that is used internally by Akka as a separator.
|
Event Sourced Entity’s Effect API
The Event Sourced Entity’s Effect defines the operations that Akka should perform when an incoming command is handled by an Event Sourced Entity.
An Event Sourced Entity Effect can either:
-
persist events and send a reply to the caller
-
directly reply to the caller if the command is not requesting any state change
-
instruct Akka to delete the entity and send a reply to the caller
-
return an error message
See also Declarative Effects for more information.
Implementing behavior
Now that we have our Entity state defined along with its events, the remaining steps can be summarized as follows:
-
declare your entity and pick a component id (it needs to be unique as it will be used for sharding purposes);
-
implement how each command is handled and which event(s) it generates;
-
provide an event handler and how it updates the entity’s state.
The class signature for our shopping cart entity will look like this:
@ComponentId("shopping-cart") (2)
public class ShoppingCartEntity extends EventSourcedEntity<ShoppingCart, ShoppingCartEvent> { (1)
}
1 | Create a class that extends EventSourcedEntity<S, E> , where S is the state type this entity will store (i.e. ShoppingCart ) and E is the top type for the events it persists (i.e. ShoppingCartEvent ). |
2 | Make sure to annotate such class with @ComponentId and pass a stable unique identifier for this entity type. |
The @ComponentId value shopping-cart is common for all instances of this entity but must be stable - cannot be changed after a production deploy - and unique across the different entity types in the service.
|
Updating state
Having created the basis of our entity, we will now define how each command is handled. In the example below, we define a method that will add a new line item to a given shopping cart. It returns an Effect
to persist an event and then sends a reply once the event is stored successfully. The state is updated by the event handler.
The only way for a command handler to modify the Entity’s state is by persisting an event. Any modifications made directly to the state (or instance variables) from the command handler are not persisted. When the Entity is passivated and reloaded, those modifications will not be present. |
public Effect<Done> addItem(LineItem item) {
if (currentState().checkedOut())
return effects().error("Cart is already checked out.");
if (item.quantity() <= 0) { (1)
return effects().error("Quantity for item " + item.productId() + " must be greater than zero.");
}
var event = new ShoppingCartEvent.ItemAdded(item); (2)
return effects()
.persist(event) (3)
.thenReply(newState -> Done.getInstance()); (4)
}
@Override
public ShoppingCart applyEvent(ShoppingCartEvent event) {
return switch (event) {
case ShoppingCartEvent.ItemAdded evt -> currentState().onItemAdded(evt); (5)
};
}
1 | The validation ensures the quantity of items added is greater than zero and it fails for calls with illegal values by returning an Effect with effects().error . |
2 | From the current incoming LineItem we create a new ItemAdded event representing the change of the cart. |
3 | We store the event by returning an Effect with effects().persist . |
4 | The acknowledgment that the command was successfully processed is only sent if the event was successfully stored and applied, otherwise there will be an error reply. The lambda parameter newState gives us access to the new state returned by applying such event. |
5 | Event handler returns the updated state after applying the event - the logic for state transition is defined inside the ShoppingCart domain model. |
As mentioned above, the business logic that allows us to transition between states was placed on the domain model as seen below:
public ShoppingCart onItemAdded(ShoppingCartEvent.ItemAdded itemAdded) {
var item = itemAdded.item();
var lineItem = updateItem(item, this); (1)
List<LineItem> lineItems = removeItemByProductId(this, item.productId()); (2)
lineItems.add(lineItem); (3)
lineItems.sort(Comparator.comparing(LineItem::productId));
return new ShoppingCart(cartId, lineItems, checkedOut); (4)
}
private static LineItem updateItem(LineItem item, ShoppingCart cart) {
return cart.findItemByProductId(item.productId())
.map(li -> li.withQuantity(li.quantity() + item.quantity()))
.orElse(item);
}
private static List<LineItem> removeItemByProductId(ShoppingCart cart, String productId) {
return cart.items().stream()
.filter(lineItem -> !lineItem.productId().equals(productId))
.collect(Collectors.toList());
}
public Optional<LineItem> findItemByProductId(String productId) {
Predicate<LineItem> lineItemExists =
lineItem -> lineItem.productId().equals(productId);
return items.stream().filter(lineItemExists).findFirst();
}
1 | For an existing item, we will make sure to sum the existing quantity with the incoming one. |
2 | Returns an update list of items without the existing item. |
3 | Adds the update item to the shopping cart. |
4 | Returns a new instance of the shopping cart with the updated line items. |
Retrieving state
To have access to the current state of the entity we can use currentState()
as you have probably noticed from the examples above. However, what if this is the first command we are receiving for this entity? The following example shows the implementation of the read-only command handler getCart
:
private final String entityId;
public ShoppingCartEntity(EventSourcedEntityContext context) {
this.entityId = context.entityId(); (1)
}
@Override
public ShoppingCart emptyState() { (2)
return new ShoppingCart(entityId, Collections.emptyList(), false);
}
public ReadOnlyEffect<ShoppingCart> getCart() {
return effects().reply(currentState()); (3)
}
1 | Stores the entityId on an internal attribute so we can use it later. |
2 | Provides initial state - overriding emptyState() is optional but if not doing it, be careful to deal with a currentState() with a null value when receiving the first command or event. |
3 | Returns the current state as reply for the request. |
We are returning the internal state directly back to the requester. In the endpoint, it’s usually best to convert this internal domain model into a public model so the internal representation is free to evolve without breaking clients code. |
Snapshots
Snapshots are an important optimization for Event Sourced Entities that persist many events. Rather than reading the entire journal upon loading or restart, Akka can initiate them from a snapshot.
Snapshots are stored and handled automatically by Akka without any specific code required. Snapshots are stored after a configured number of events:
akka.javasdk.event-sourced-entity.snapshot-every = 100
When the Event Sourced Entity is loaded again, the snapshot will be loaded before any other events are received.
Deleting an Entity
Normally, Event Sourced Entities are not deleted because the history of the events typically provide business value. For certain use cases or for regulatory reasons the entity can be deleted.
public Effect<Done> checkout() {
if (currentState().checkedOut())
return effects().reply(Done.getInstance());
return effects()
.persist(new ShoppingCartEvent.CheckedOut()) (1)
.deleteEntity() (2)
.thenReply(newState -> Done.getInstance());
}
1 | Persist final event before deletion, which is handled as any other event. |
2 | Instruction to delete the entity. |
When you give the instruction to delete the entity it will still exist for some time, including its events and snapshots. The actual removal of events and snapshots will be deleted later to give downstream consumers time to process all prior events, including the final event that was persisted together with the deleteEntity
effect. By default, the existence of the entity is completely cleaned up after a week.
It is not allowed to persist more events after the entity has been "marked" as deleted. You can still handle read requests to the entity until it has been completely removed.
It is best to not reuse the same entity id after deletion, but if that happens after the entity has been completely removed it will be instantiated as a completely new entity without any knowledge of previous state.
Note that deleting View state must be handled explicitly.
Multi-region replication
Stateful components like Event Sourced Entities or Workflow can be replicated to other regions. This is useful for several reasons:
-
resilience to tolerate failures in one location and still be operational, even multi-cloud redundancy
-
possibility to serve requests from a location near the user to provide better responsiveness
-
load balancing to be able to handle high throughput
For each stateful component instance there is a primary region, which handles all write requests. Read requests can be served from any region.
Read requests are defined by declaring the command handler method with ReadOnlyEffect
as return type. A read-only handler cannot update the state, and that is enforced at compile time.
public ReadOnlyEffect<ShoppingCart> getCart() {
return effects().reply(currentState()); (3)
}
Write requests are defined by declaring the command handler method with Effect
as return type, instead of ReadOnlyEffect
. Write requests are routed to the primary region and handled by the stateful component instance in that region even if the original call to the instance with the component client was made from another region.
State changes (Workflow) or events (Event Sourced Entity) persisted by the instance in the primary region are replicated to other regions and processed by corresponding instance there. This means that the state of the stateful components in all regions are updated from the primary.
The replication is asynchronous, which means that read replicas are eventually updated. Normally within a few milliseconds, but if there is for example a problem with the network between the regions it can take longer time for the read replicas to become up to date, but eventually they will.
This also means that you might not see your own writes, immediately. Consider the following:
-
send a write request and that is routed to a primary in another region
-
after receiving the response of the write request, you send a read request that is served by the non-primary region
-
the stateful component instance in the non-primary region might not have seen the replicated changes yet, and therefore replies with "stale" information
If it’s important for some read requests to have seen latest writes you can use Effect
for such command handler, even though it is not persisting any events. Then the request will be routed to the primary and use the latest fully consistent state.
The operational aspects are described in Regions.
Side Effects
An entity doesn’t perform any external side effects aside from persisting events and replying to the request. Side effects can be handled from the Workflow, Consumer, or Endpoint components that are calling the entity.
Testing the Entity
There are two ways to test an Entity:
-
Unit test, which only runs the Entity component with a test kit.
-
Integration test, running the entire service with a test kit and the test interacting with it using a component client or over HTTP requests.
Each way has its benefits, unit tests are faster and provide more immediate feedback about success or failure but can only test a single entity at a time and in isolation. Integration tests, on the other hand, are more realistic and allow many entities to interact with other components inside and outside the service.
Unit tests
The following snippet shows how the EventSourcedTestKit
is used to test the ShoppingCartEntity
implementation. Akka provides two main APIs for unit tests, the EventSourcedTestKit
and the EventSourcedResult
. The former gives us the overall state of the entity and all the events produced by all the calls to the Entity. While the latter only holds the effects produced for each individual call to the Entity.
package shoppingcart.application;
import akka.Done;
import akka.javasdk.testkit.EventSourcedTestKit;
import org.junit.jupiter.api.Test;
import shoppingcart.domain.ShoppingCart;
import java.util.List;
import static shoppingcart.domain.ShoppingCartEvent.ItemAdded;
import static org.junit.jupiter.api.Assertions.assertEquals;
public class ShoppingCartTest {
private final ShoppingCart.LineItem akkaTshirt = new ShoppingCart.LineItem("akka-tshirt", "Akka Tshirt", 10);
@Test
public void testAddLineItem() {
var testKit = EventSourcedTestKit.of(ShoppingCartEntity::new); (1)
{
var result = testKit.call(e -> e.addItem(akkaTshirt)); (2)
assertEquals(Done.getInstance(), result.getReply()); (3)
var itemAdded = result.getNextEventOfType(ItemAdded.class);
assertEquals(10, itemAdded.item().quantity()); (4)
}
// actually we want more akka tshirts
{
var result = testKit.call(e -> e.addItem(akkaTshirt.withQuantity(5))); (5)
assertEquals(Done.getInstance(), result.getReply());
var itemAdded = result.getNextEventOfType(ItemAdded.class);
assertEquals(5, itemAdded.item().quantity());
}
{
assertEquals(testKit.getAllEvents().size(), 2); (6)
var result = testKit.call(ShoppingCartEntity::getCart); (7)
assertEquals(
new ShoppingCart("testkit-entity-id", List.of(akkaTshirt.withQuantity(15)), false),
result.getReply());
}
}
}
1 | Creates the TestKit passing the constructor of the Entity. |
2 | Calls the method addItem from the Entity in the EventSourcedTestKit with quantity 10 . |
3 | Asserts the return value is "OK" . |
4 | Returns the next event of type IdemAdded and asserts on the quantity. |
5 | Add a new item with quantity 5 . |
6 | Asserts that the total number of events should be 2. |
7 | Calls the getCart method and asserts that quantity should be 15 . |
The EventSourcedTestKit is stateful, and it holds the state of a single entity instance in memory. If you want to test more than one entity in a test, you need to create multiple instance of EventSourcedTestKit .
|
EventSourcedResult
Calling a command handler through the TestKit gives us back an EventSourcedResult
. This class has methods that we can use to assert the handling of the command, such as:
-
getReply()
- the response from the command handler if there was one, if not an, exception is thrown, failing the test. -
getAllEvents()
- all the events persisted by handling the command. -
getState()
- the state of the entity after applying any events the command handler persisted. -
getNextEventOfType(ExpectedEvent.class)
- check the next of the persisted events against an event type, return it for inspection if it matches, or fail the test if it does not. The event gets consumed once is inspected and the next call will look for a subsequent event.
EventSourcedTestKit
For the above example, this class provides access to all the command handlers of the ShoppingCart
entity for unit testing. In addition to that also has the following methods:
-
getState()
- the current state of the entity, it is updated on each method call persisting events. -
getAllEvents()
- all events persisted since the creation of the testkit instance.
Integration tests
The skeleton of an Integration Test is generated for you if you use the archetype to start your Akka service. Let’s see what it could look like to test our ShoppingCartEntity
:
public class IntegrationTest extends TestKitSupport { (1)
@Test
public void createAndManageCart() {
String cartId = "card-abc";
var item1 = new LineItem("tv", "Super TV 55'", 1);
var response1 = await(
componentClient (2)
.forEventSourcedEntity(cartId) (3)
.method(ShoppingCartEntity::addItem) (4)
.invokeAsync(item1)
);
Assertions.assertNotNull(response1);
// confirming only one product remains
ShoppingCart cartUpdated = await(
componentClient
.forEventSourcedEntity(cartId)
.method(ShoppingCartEntity::getCart) (5)
.invokeAsync()
);
Assertions.assertEquals(1, cartUpdated.items().size()); (6)
Assertions.assertEquals(item2, cartUpdated.items().get(0));
}
}
1 | Note the test class must extend TestKitSupport . |
2 | A built-in component client is provided to interact with the components. |
3 | Request to create a new shopping cart with id cart-abc . |
4 | Request to add an item to the cart. |
5 | Request to retrieve current status of the shopping cart. |
6 | Assert there should only be one item. |
The integration tests in samples are under a specific project profile it and can be run using mvn verify -Pit .
|
Exposing entities directly
You would normally not expose Entities, Workflows and Views directly to the outside world, but have an Endpoint or Consumer in front of it.
However, a built-in HTTP endpoint for the component can be enabled. The primary purpose of this component endpoint is for convenience in local development so that you can try the entity with curl
without having to define the final HttpEndpoint
.
Enable access
To enable the component’s HTTP endpoint you can disable access control checks in application.conf
:
akka.javasdk.dev-mode.acl.enabled = false
Alternatively, start with:
mvn compile exec:java -Dakka.javasdk.dev-mode.acl.enabled=false
Note that this acl
configuration is only used when running locally and not in production. To enable component endpoints in production you must add @Acl
to the component class:
// Opened up for access from the public internet.
// For actual services meant for production this must be carefully considered,
// and often set more limited
@Acl(allow = @Acl.Matcher(principal = Acl.Principal.INTERNET))
API
The entity is exposed at a fixed path:
/akka/v1.0/entity/<component id>/<entity id>/<method>
In our shopping cart example that is:
curl localhost:9000/akka/v1.0/entity/carts/12345/getCart
or, to add an item:
curl localhost:9000/akka/v1.0/entity/carts/12345/addItem \
--header "Content-Type: application/json" \
-XPOST \
--data '{"productId":"akka-tshirt","name":"Akka Tshirt","quantity":10}'