Setup your application
Add the Akka Projections core library to a new project. This isn’t strictly required, because as we add other dependencies in the following steps it will transitively include core as a dependency, but it never hurts to be explicit.
The Akka dependencies are available from Akka’s library repository. To access them there, you need to configure the URL for this repository.
Additionally, add the dependency as below.
Define the event type protocol that will represent each Envelope
streamed from the Source Provider. Add ShoppingCartEvents
to your project:
- Scala
- Java
-
source
package jdocs.guide; import akka.serialization.jackson.CborSerializable; import java.time.Instant; public class ShoppingCartEvents { public interface Event extends CborSerializable { String getCartId(); } public interface ItemEvent extends Event { String getItemId(); } public static final class ItemAdded implements ItemEvent { public final String cartId; public final String itemId; public final int quantity; public ItemAdded(String cartId, String itemId, int quantity) { this.cartId = cartId; this.itemId = itemId; this.quantity = quantity; } public String getCartId() { return this.cartId; } public String getItemId() { return this.itemId; } } public static final class ItemRemoved implements ItemEvent { public final String cartId; public final String itemId; public final int oldQuantity; public ItemRemoved(String cartId, String itemId, int oldQuantity) { this.cartId = cartId; this.itemId = itemId; this.oldQuantity = oldQuantity; } public String getCartId() { return this.cartId; } public String getItemId() { return this.itemId; } } public static final class ItemQuantityAdjusted implements ItemEvent { public final String cartId; public final String itemId; public final int newQuantity; public final int oldQuantity; public ItemQuantityAdjusted(String cartId, String itemId, int newQuantity, int oldQuantity) { this.cartId = cartId; this.itemId = itemId; this.newQuantity = newQuantity; this.oldQuantity = oldQuantity; } public String getCartId() { return this.cartId; } public String getItemId() { return this.itemId; } } public static final class CheckedOut implements Event { public final String cartId; public final Instant eventTime; public CheckedOut(String cartId, Instant eventTime) { this.cartId = cartId; this.eventTime = eventTime; } public String getCartId() { return this.cartId; } } }
To enable serialization and deserialization of events with Akka Persistence it’s necessary to define a base type for your event type hierarchy. In this guide we are using akka.serialization.jackson.CborSerializable
that has a built-in binding to Jackson Serialization.
For Jackson serialization to work correctly in Java projects you must use the javac
compiler parameter -parameters
when building your project. In maven you can add an argument to maven-compiler-plugin
plugin under compilerArgs
(see an example here).
Define the persistence tags to be used in your project. Note that partitioned tags will be used later when running the projection in Akka Cluster. Add ShoppingCartTags
to your project:
- Scala
- Java
-
source
package jdocs.guide; public class ShoppingCartTags { public static String SINGLE = "shopping-cart"; public static String[] TAGS = {"carts-0", "carts-1", "carts-2"}; }
Create the ShoppingCartApp
with an akka.actor.typed.ActorSystem
(API: ActorSystem
) for Projections to use. Create an empty Guardian Actor (the root Actor of the ActorSystem
). We will populate this Actor in the following steps of the guide. Note that we are using the jdocs.scaladsl
package. You may use any package, but we include this package in snippets throughout the guide.
- Scala
- Java
-
source
package jdocs.guide; import akka.actor.typed.ActorSystem; import akka.actor.typed.javadsl.Behaviors; import akka.projection.ProjectionBehavior; import akka.projection.eventsourced.EventEnvelope; import com.typesafe.config.Config; import com.typesafe.config.ConfigFactory; public class ShoppingCartApp { public static void main(String[] args) throws Exception { Config config = ConfigFactory.load("guide-shopping-cart-app.conf"); ActorSystem.create( Behaviors.setup( context -> { ActorSystem<Void> system = context.getSystem(); // ... return Behaviors.empty(); }), "ShoppingCartApp", config); } }