User guide

This is a brief user guide to Eventuate. It is recommended to read sections Overview and Architecture first. Based on simple examples, you’ll see how to

  • implement an event-sourced actor
  • replicate actor state with event sourcing
  • detect concurrent updates to replicated state
  • track conflicts from concurrent updates
  • resolve conflicts automatically and interactively
  • make concurrent updates conflict-free with operation-based CRDTs
  • implement an event-sourced view over many event-sourced actors
  • achieve causal read consistency across event-sourced actors and views and
  • implement event-driven communication between event-sourced actors.

The user guide only scratches the surface of Eventuate. You can find further details in the Reference.

Event-sourced actors

An event-sourced actor is an actor that captures changes to its internal state as a sequence of events. It persists these events to an event log and replays them to recover internal state after a crash or a planned re-start. This is the basic idea behind event sourcing: instead of storing current application state, the full history of changes is stored as immutable facts and current state is derived from these facts.

Event-sourced actors distinguish between commands and events. During command processing they usually validate external commands against internal state and, if validation succeeds, write one or more events to their event log. During event processing they consume events they have written and update internal state by handling these events.

Hint

Event-sourced actors can also write new events during event processing. This is covered in section Event-driven communication.

Concrete event-sourced actors must implement the EventsourcedActor trait. The following ExampleActor maintains state of type Vector[String] to which entries can be appended:

  • scala notranslate
  • java notranslate
import scala.util._
import akka.actor._
import com.rbmhtechnology.eventuate.EventsourcedActor

// Commands
case object Print
case class Append(entry: String)

// Command replies
case class AppendSuccess(entry: String)
case class AppendFailure(cause: Throwable)

// Event
case class Appended(entry: String)

class ExampleActor(override val id: String,
                   override val aggregateId: Option[String],
                   override val eventLog: ActorRef) extends EventsourcedActor {

  private var currentState: Vector[String] = Vector.empty

  override def onCommand = {
    case Print =>
      println(s"[id = $id, aggregate id = ${aggregateId.getOrElse("<undefined>")}] ${currentState.mkString(",")}")
    case Append(entry) => persist(Appended(entry)) {
      case Success(evt) => sender() ! AppendSuccess(entry)
      case Failure(err) => sender() ! AppendFailure(err)
    }
  }

  override def onEvent = {
    case Appended(entry) => currentState = currentState :+ entry
  }
}

For modifying currentState, applications send Append commands which are handled by the onCommand handler. From an Append command, the handler derives an Appended event and persists it to the given eventLog. If persistence succeeds, the command sender is informed about successful processing. If persistence fails, the command sender is informed about the failure so it can retry, if needed.

The onEvent handler updates currentState from persisted events and is automatically called after a successful persist. If the actor is re-started, either after a crash or during normal application start, persisted events are replayed to onEvent which recovers internal state before new commands are processed.

EventsourcedActor implementations must define a global unique id and require an eventLog actor reference for writing and replaying events. An event-sourced actor may also define an optional aggregateId which has an impact how events are routed between event-sourced actors.

Hint

Section Event log explains how to create eventLog actor references.

Creating a single instance

In the following, a single instance of ExampleActor is created and two Append commands are sent to it:

  • scala notranslate
  • java notranslate
val system: ActorSystem = // ...
val eventLog: ActorRef = // ...

val ea1 = system.actorOf(Props(new ExampleActor("1", Some("a"), eventLog)))

ea1 ! Append("a")
ea1 ! Append("b")

Sending a Print command

  • scala notranslate
  • java notranslate
ea1 ! Print

should print:

[id = 1, aggregate id = a] a,b

When the application is re-started, persisted events are replayed to onEvent which recovers currentState. Sending another Print command should print again:

[id = 1, aggregate id = a] a,b

Note

In the following sections, several instances of ExampleActor are created. It is assumed that they share a Replicated event log and are running at different locations.

A shared event log is a pre-requisite for event-sourced actors to consume each other’s events. However, sharing an event log doesn’t necessarily mean broadcast communication between all actors on the same log. It is the aggreagteId that determines which actors consume each other’s events.

Creating two isolated instances

When creating two instances of ExampleActor with different aggregateIds, they are isolated from each other, by default, and do not consume each other’s events:

  • scala notranslate
  • java notranslate
val b2 = system.actorOf(Props(new ExampleActor("2", Some("b"), eventLog)))
val c3 = system.actorOf(Props(new ExampleActor("3", Some("c"), eventLog)))

b2 ! Append("a")
b2 ! Append("b")

c3 ! Append("x")
c3 ! Append("y")

Sending two Print commands

  • scala notranslate
  • java notranslate
b2 ! Print
c3 ! Print

should print:

[id = 2, aggregate id = b] a,b
[id = 3, aggregate id = c] x,y

Creating two replica instances

When creating two ExampleActor instances with the same aggregateId, they consume each other’s events [1].

  • scala notranslate
  • java notranslate
// created at location 1
val d4 = system.actorOf(Props(new ExampleActor("4", Some("d"), eventLog)))

// created at location 2
val d5 = system.actorOf(Props(new ExampleActor("5", Some("d"), eventLog)))

d4 ! Append("a")

Here, d4 processes an Append command and persists an Appended event. Both, d4 and d5, consume that event and update their internal state. After waiting a bit for convergence, sending a Print command to both actors should print:

[id = 4, aggregate id = d] a
[id = 5, aggregate id = d] a

After both replicas have converged, another Append is sent to d5.

  • scala notranslate
  • java notranslate
d5 ! Append("b")

Again both actors consume the event and sending another Print command should print:

[id = 4, aggregate id = d] a,b
[id = 5, aggregate id = d] a,b

Warning

As you have probably recognized, replica convergence in this example can only be achieved if the second Append command is sent after both actors have processed the Appended event from the first Append command.

In other words, the first Appended event must happen before the second one. Only in this case, these two events can have a causal relationship. Since events are guaranteed to be delivered in potential causal order to all replicas, they can converge to the same state.

When concurrent updates are made to both replicas, the corresponding Appended events are not causally related and can be delivered in any order to both replicas. This may cause replicas to diverge because append operations do not commute. The following sections give examples how to detect and handle concurrent updates.

Detecting concurrent updates

Eventuate tracks happened-before relationships (= potential causality) of events with Vector clocks. Why is that needed at all? Let’s assume that an event-sourced actor emits an event e1 for changing internal state and later receives an event e2 from a replica instance. If the replica instance emits e2 after having processed e1, the actor can apply e2 as regular update. If the replica instance emits e2 before having received e1, the actor receives a concurrent, potentially conflicting event.

How can the actor determine if e2 is a regular i.e. causally related or concurrent update? It can do so by comparing the vector timestamps of e1 and e2, where t1 is the vector timestamp of e1 and t2 the vector timestamp of e2. If events e1 and e2 are concurrent then t1 conc t2 evaluates to true. Otherwise, they are causally related and t1 < t2 evaluates to true (because e1 happened-before e2).

The vector timestamp of an event can be obtained with lastVectorTimestamp during event processing. Vector timestamps can be attached as update timestamp to current state and compared with the vector timestamp of a new event in order to determine whether the new event is causally related to the previous state update or not[2]:

  • scala notranslate
  • java notranslate
import com.rbmhtechnology.eventuate.EventsourcedActor
import com.rbmhtechnology.eventuate.VectorTime

class ExampleActor(override val id: String,
                   override val aggregateId: Option[String],
                   override val eventLog: ActorRef) extends EventsourcedActor {

  private var currentState: Vector[String] = Vector.empty
  private var updateTimestamp: VectorTime = VectorTime()

  override def onCommand = {
    // ...
  }

  override def onEvent = {
    case Appended(entry) =>
      if (updateTimestamp < lastVectorTimestamp) {
        // regular update
        currentState = currentState :+ entry
        updateTimestamp = lastVectorTimestamp
      } else if (updateTimestamp conc lastVectorTimestamp) {
        // concurrent update
        // TODO: track conflicting versions
      }
  }
}

Attaching update timestamps to current state and comparing them with vector timestamps of new events can be easily abstracted over so that applications don’t have to deal with these low level details, as shown in the next section.

Tracking conflicting versions

If state update operations from concurrent events do not commute, conflicting versions of actor state arise that must be tracked and resolved. This can be done with Eventuate’s ConcurrentVersions[S, A] abstraction and an application-defined update function of type (S, A) => S where S is the type of actor state and A the update type. In our example, the ConcurrentVersions type is ConcurrentVersions[Vector[String], String] and the update function (s, a) => s :+ a:

  • scala notranslate
  • java notranslate
import scala.collection.immutable.Seq
import com.rbmhtechnology.eventuate.{ConcurrentVersions, Versioned}
import com.rbmhtechnology.eventuate.EventsourcedActor

class ExampleActor(override val id: String,
                   override val aggregateId: Option[String],
                   override val eventLog: ActorRef) extends EventsourcedActor {

  private var versionedState: ConcurrentVersions[Vector[String], String] =
    ConcurrentVersions(Vector.empty, (s, a) => s :+ a)

  override def onCommand = {
    // ...
  }

  override def onEvent = {
    case Appended(entry) =>
      versionedState = versionedState.update(entry, lastVectorTimestamp)
      if (versionedState.conflict) {
        val conflictingVersions: Seq[Versioned[Vector[String]]] = versionedState.all
        // TODO: resolve conflicting versions
      } else {
        val currentState: Vector[String] = versionedState.all.head.value
        // ...
      }
  }
}

Internally, ConcurrentVersions maintains versions of actor state in a tree structure where each concurrent update creates a new branch. The shape of the tree is determined solely by the vector timestamps of the corresponding update events.

An event’s vector timestamp is passed as lastVectorTimestamp argument to update. The update method internally creates a new version by applying the update function (s, a) => s :+ a to the closest predecessor version and the actual update value (entry). The lastVectorTimestamp is attached as update timestamp to the newly created version.

Concurrent versions of actor state and their update timestamp can be obtained with all which is a sequence of type Seq[Versioned[Vector[String]]] in our example. The Versioned data type represents a particular version of actor state and its update timestamp (= vectorTimestamp field).

If all contains only a single element, there is no conflict and the element represents the current, conflict-free actor state. If the sequence contains two or more elements, there is a conflict where the elements represent conflicting versions of actor states. They can be resolved either automatically or interactively.

Note

Only concurrent updates to replicas with the same aggregateId may conflict. Concurrent updates to actors with different aggregateId do not conflict (unless an application does custom Event routing).

Also, if the data type of actor state is designed in a way that update operations commute, concurrent updates can be made conflict-free. This is discussed in section Operation-based CRDTs.

Resolving conflicting versions

Automated conflict resolution

The following is a simple example of automated conflict resolution: if a conflict has been detected, the version with the higher wall clock timestamp is selected to be the winner. In case of equal wall clock timestamps, the version with the lower emitter id is selected. The wall clock timestamp can be obtained with lastSystemTimestamp during event handling, the emitter id with lastEmitterId. The emitter id is the id of the EventsourcedActor that emitted the event.

  • scala notranslate
  • java notranslate
class ExampleActor(override val id: String,
                   override val aggregateId: Option[String],
                   override val eventLog: ActorRef) extends EventsourcedActor {

  private var versionedState: ConcurrentVersions[Vector[String], String] =
    ConcurrentVersions(Vector.empty, (s, a) => s :+ a)

  override def onCommand = {
    // ...
  }

  override def onEvent = {
    case Appended(entry) =>
      versionedState = versionedState
        .update(entry, lastVectorTimestamp, lastSystemTimestamp, lastEmitterId)
      if (versionedState.conflict) {
        val conflictingVersions = versionedState.all.sortWith { (v1, v2) =>
          if (v1.systemTimestamp == v2.systemTimestamp) v1.creator < v2.creator
          else v1.systemTimestamp > v2.systemTimestamp
        }
        val winnerTimestamp: VectorTime = conflictingVersions.head.vectorTimestamp
        versionedState = versionedState.resolve(winnerTimestamp)
      }
  }
}

Here, conflicting versions are sorted by descending wall clock timestamp and ascending emitter id where the latter is tracked as creator of the version. The first version is selected to be the winner. Its vector timestamp is passed as argument to resolve which selects this version and discards all other versions.

More advanced conflict resolution could select a winner depending on the actual value of concurrent versions. After selection, an application could even update the winner with the merged value of all conflicting versions[3].

Note

For replicas to converge, it is important that winner selection does not depend on the order of conflicting events. In our example, this is the case because wall clock timestamp and emitter id comparison is transitive.

Interactive conflict resolution

Interactive conflict resolution does not resolve conflicts immediately but requests the user to inspect and resolve a conflict. The following is a very simple example of interactive conflict resolution: a user selects a winner version if conflicting versions of application state exist.

  • scala notranslate
  • java notranslate
case class Append(entry: String)
case class AppendRejected(entry: String, conflictingVersions: Seq[Versioned[Vector[String]]])

case class Resolve(selectedTimestamp: VectorTime)
case class Resolved(selectedTimestamp: VectorTime)

class ExampleActor(override val id: String,
                   override val aggregateId: Option[String],
                   override val eventLog: ActorRef) extends EventsourcedActor {

  private var versionedState: ConcurrentVersions[Vector[String], String] =
    ConcurrentVersions(Vector.empty, (s, a) => s :+ a)

  override def onCommand = {
    case Append(entry) if versionedState.conflict =>
      sender() ! AppendRejected(entry, versionedState.all)
    case Append(entry) =>
      // ...
    case Resolve(selectedTimestamp) => persist(Resolved(selectedTimestamp)) {
      case Success(evt) => // reply to sender omitted ...
      case Failure(err) => // reply to sender omitted ...
    }
  }

  override def onEvent = {
    case Appended(entry) =>
      versionedState = versionedState
        .update(entry, lastVectorTimestamp, lastSystemTimestamp, lastEmitterId)
    case Resolved(selectedTimestamp) =>
      versionedState = versionedState.resolve(selectedTimestamp, lastVectorTimestamp)
  }
}

When a user tries to Append in presence of a conflict, the ExampleActor rejects the update and requests the user to select a winner version from a sequence of conflicting versions. The user then sends the update timestamp of the winner version as selectedTimestamp with a Resolve command from which a Resolved event is derived and persisted. Handling of Resolved at all replicas finally resolves the conflict.

In addition to just selecting a winner, an application could also update the winner version in a second step, for example, with a value derived from the merge result of conflicting versions. Support for atomic, interactive conflict resolution with an application-defined merge function is planned for later Eventuate releases.

Note

Interactive conflict resolution requires agreement among replicas that are affected by a given conflict: only one of them may emit the Resolved event. This does not necessarily mean distributed lock acquisition or leader (= resolver) election but can also rely on static rules such as only the initial creator location of an aggregate is allowed to resolve the conflict[4]. This rule is implemented in the Example application.

Operation-based CRDTs

If state update operations commute, there’s no need to use Eventuate’s ConcurrentVersions utility. A simple example is a replicated counter, which converges because its increment and decrement operations commute.

A formal to approach to commutative replicated data types (CmRDTs) or operation-based CRDTs is given in the paper A comprehensive study of Convergent and Commutative Replicated Data Types by Marc Shapiro et al. Eventuate is a good basis for implementing operation-based CRDTs:

  • Update operations can be modeled as events and reliably broadcasted to all replicas by a Replicated event log.
  • The command and event handler of an event-sourced actor can be used to implement the two update phases mentioned in the paper: atSource and downstream, respectively.
  • All downstream preconditions mentioned in the paper are satisfied in case of causal delivery of update operations which is guaranteed for actors consuming from a replicated event log.

Eventuate currently implements 5 out of 12 operation-based CRDTs specified in the paper. These are Counter, MV-Register, LWW-Register, OR-Set and OR-Cart (a shopping cart CRDT). They can be instantiated and used via their corresponding CRDT services. CRDT operations are asynchronous methods on the service interfaces. CRDT services free applications from dealing with low-level details like event-sourced actors or command messages directly. The following is the definition of ORSetService:

  • scala notranslate
  • java notranslate
/**
 * Replicated [[ORSet]] CRDT service.
 *
 * @param serviceId Unique id of this service.
 * @param log Event log.
 * @tparam A [[ORSet]] entry type.
 */
class ORSetService[A](val serviceId: String, val log: ActorRef)(implicit val system: ActorSystem, val ops: CRDTServiceOps[ORSet[A], Set[A]])
  extends CRDTService[ORSet[A], Set[A]] {

  /**
   * Adds `entry` to the OR-Set identified by `id` and returns the updated entry set.
   */
  def add(id: String, entry: A): Future[Set[A]] =
    op(id, AddOp(entry))

  /**
   * Removes `entry` from the OR-Set identified by `id` and returns the updated entry set.
   */
  def remove(id: String, entry: A): Future[Set[A]] =
    op(id, RemoveOp(entry))

  start()
}

/**
 * Persistent add operation used for [[ORSet]] and [[ORCart]].
 */
case class AddOp(entry: Any) extends CRDTFormat

/**
 * Persistent remove operation used for [[ORSet]] and [[ORCart]].
 */
case class RemoveOp(entry: Any, timestamps: Set[VectorTime] = Set.empty) extends CRDTFormat

The ORSetService is a CRDT service that manages ORSet instances. It implements the asynchronous add and remove methods and inherits the value(id: String): Future[Set[A]] method from CRDTService[ORSet[A], Set[A]] for reading the current value. Their id parameter identifies an ORSet instance. Instances are automatically created by the service on demand. A usage example is the ReplicatedOrSetSpec that is based on Akka’s multi node testkit.

A CRDT service also implements a save(id: String): Future[SnapshotMetadata] method for saving CRDT snapshots. Snapshots may reduce recovery times of CRDTs with a long update history but are not required for CRDT persistence.

New operation-based CRDTs and their corresponding services can be developed with the CRDT development framework, by defining an instance of the CRDTServiceOps type class and implementing the CRDTService trait. Take a look at the CRDT sources for examples.

Hint

Eventuate’s CRDT approach is also described in this article.

Event-sourced views

Event-sourced views are a functional subset of event-sourced actors. They can only consume events from an event log but cannot produce new events. Concrete event-sourced views must implement the EventsourcedView trait. In the following example, the view counts all Appended and Resolved events emitted by all event-sourced actors to the same eventLog:

  • scala notranslate
  • java notranslate
import akka.actor.ActorRef
import com.rbmhtechnology.eventuate.EventsourcedView
import com.rbmhtechnology.eventuate.VectorTime

case class Appended(entry: String)
case class Resolved(selectedTimestamp: VectorTime)

case object GetAppendCount
case class GetAppendCountReply(count: Long)

case object GetResolveCount
case class GetResolveCountReply(count: Long)

class ExampleView(override val id: String, override val eventLog: ActorRef) extends EventsourcedView {
  private var appendCount: Long = 0L
  private var resolveCount: Long = 0L

  override def onCommand = {
    case GetAppendCount => sender() ! GetAppendCountReply(appendCount)
    case GetResolveCount => sender() ! GetResolveCountReply(resolveCount)
  }

  override def onEvent = {
    case Appended(_) => appendCount += 1L
    case Resolved(_) => resolveCount += 1L
  }
}

Event-sourced views handle events in the same way as event-sourced actors by implementing an onEvent handler. The onCommand handler in the example processes the queries GetAppendCount and GetResolveCount.

ExampleView implements the mandatory global unique id but doesn’t define an aggregateId. A view that doesn’t define an aggregateId can consume events from all event-sourced actors on the same event log. If it defines an aggregateId it can only consume events from event-sourced actors with the same aggregateId (assuming the default Event routing rules).

Hint

While event-sourced views maintain view state in-memory, Event-sourced writers can be used to persist view state to external databases. A specialization of event-sourced writers are Event-sourced processors whose external database is an event log.

Conditional requests

Causal read consistency is the default when reading state from a single event-sourced actor or view. The event stream received by that actor is always causally ordered, hence, it will never see an effect before having seen its cause.

The situation is different when a client reads from multiple actors. Imagine two event-sourced actor replicas where a client updates one replica and observes the updated state with the reply. A subsequent from the other replica, made by the same client, may return the old state which violates causal consistency.

Similar considerations can be made for reading from an event-sourced view after having made an update to an event-sourced actor. For example, an application that successfully appended an entry to ExampleActor may not immediately see that update in the appendCount of ExampleView. To achieve causal read consistency, the view should delay command processing until the emitted event has been consumed by the view. This can be achieved with a ConditionalRequest.

  • scala notranslate
  • java notranslate
import scala.concurrent.duration._
import scala.util._
import akka.actor._
import akka.pattern.ask
import akka.util.Timeout
import com.rbmhtechnology.eventuate._

case class Append(entry: String)
case class AppendSuccess(entry: String, updateTimestamp: VectorTime)

class ExampleActor(override val id: String,
                   override val eventLog: ActorRef) extends EventsourcedActor {

  private var currentState: Vector[String] = Vector.empty
  override val aggregateId = Some(id)

  override def onCommand = {
    case Append(entry) => persist(Appended(entry)) {
      case Success(evt) =>
        sender() ! AppendSuccess(entry, lastVectorTimestamp)
      // ...
    }
    // ...
  }

  override def onEvent = {
    case Appended(entry) => currentState = currentState :+ entry
  }
}

class ExampleView(override val id: String, override val eventLog: ActorRef)
  extends EventsourcedView with ConditionalRequests {
  // ...
}

val ea = system.actorOf(Props(new ExampleActor("ea", eventLog)))
val ev = system.actorOf(Props(new ExampleView("ev", eventLog)))

import system.dispatcher
implicit val timeout = Timeout(5.seconds)

for {
  AppendSuccess(_, timestamp) <- ea ? Append("a")
  GetAppendCountReply(count)  <- ev ? ConditionalRequest(timestamp, GetAppendCount)
} println(s"append count = $count")

Here, the ExampleActor includes the event’s vector timestamp in its AppendSuccess reply. Together with the actual GetAppendCount command, the timestamp is included as condition in a ConditionalRequest and sent to the view. For ConditionalRequest processing, an event-sourced view must extend the ConditionalRequests trait. ConditionalRequests internally delays the command, if needed, and only dispatches GetAppendCount to the view’s onCommand handler if the condition timestamp is in the causal past of the view (which is earliest the case when the view consumed the update event). When running the example with an empty event log, it should print:

append count = 1

Note

Not only event-sourced views but also event-sourced actors, stateful event-sourced writers and processors can extend ConditionalRequests. Delaying conditional requests may re-order them relative to other conditional and non-conditional requests.

Event-driven communication

Earlier sections have already shown one form of event collaboration: state replication. For that purpose, event-sourced actors of the same type exchange their events to re-construct actor state at different locations.

In more general cases, event-sourced actors of different type exchange events to achieve a common goal. They react on received events by updating internal state and producing new events. This form of event collaboration is called event-driven communication. In the following example, two event-actors collaborate in a ping-pong game where

  • a PingActor emits a Ping event on receiving a Pong event and
  • a PongActor emits a Pong event on receiving a Ping event
  • scala notranslate
  • java notranslate
// some imports omitted ...
import com.rbmhtechnology.eventuate.EventsourcedView.Handler
import com.rbmhtechnology.eventuate.EventsourcedActor
import com.rbmhtechnology.eventuate.PersistOnEvent

case class Ping(num: Int)
case class Pong(num: Int)

class PingActor(val id: String, val eventLog: ActorRef, completion: ActorRef)
  extends EventsourcedActor with PersistOnEvent {

  override def onCommand = {
    case "serve" => persist(Ping(1))(Handler.empty)
  }

  override def onEvent = {
    case Pong(10) if !recovering => completion ! "done"
    case Pong(i)  => persistOnEvent(Ping(i + 1))
  }
}

class PongActor(val id: String, val eventLog: ActorRef)
  extends EventsourcedActor with PersistOnEvent {

  override def onCommand = {
    case _ =>
  }
  override def onEvent = {
    case Ping(i) => persistOnEvent(Pong(i))
  }
}

val pingActor = system.actorOf(Props(new PingActor("ping", eventLog, system.deadLetters)))
val pongActor = system.actorOf(Props(new PongActor("pong", eventLog)))

pingActor ! "serve"

The ping-pong game is started by sending the PingActor a ”serve” command which persists the first Ping event. This event however is not consumed by the emitter but rather by the PongActor. The PongActor reacts on the Ping event by emitting a Pong event. Other than in previous examples, the event is not emitted in the actor’s onCommand handler but in the onEvent handler. For that purpose, the actor has to mixin the PersistOnEvent trait and use the persistOnEvent method. The emitted Pong too isn’t consumed by its emitter but rather by the PingActor, emitting another Ping, and so on. The game ends when the PingActor received the 10th Pong.

Note

The ping-pong game is reliable. When an actor crashes and is re-started, the game is reliably resumed from where it was interrupted. The persistOnEvent method is idempotent i.e. no duplicates are written under failure conditions and later event replay. When deployed at different location, the ping-pong actors are also partition-tolerant. When their game is interrupted by a network partition, it is automatically resumed when the partition heals.

Furthermore, the actors don’t need to care about idempotency in their business logic i.e. they can assume to receive a de-duplicated and causally-ordered event stream in their onEvent handler. This is a significant advantage over at-least-once delivery based communication with ConfirmedDelivery, for example, which can lead to duplicates and message re-ordering.

In a more real-world example, there would be several actors of different type collaborating to achieve a common goal, for example, in a distributed business process. These actors can be considered as event-driven and event-sourced microservices, collaborating on a causally ordered event stream in a reliable and partition-tolerant way. Furthermore, when partitioned, they remain available for local writes and automatically catch up with their collaborators when the partition heals.

Hint

Further persistOnEvent details are described in the PersistOnEvent API docs.

[1]EventsourcedActors and EventsourcedViews that have an undefined aggregateId can consume events from all other actors on the same event log.
[2]Attached update timestamps are not version vectors because Eventuate uses vector clock update rules instead of version vector update rules. Consequently, update timestamp equivalence cannot be used as criterion for replica convergence.
[3]A formal approach to automatically merge concurrent versions of application state are convergent replicated data types (CvRDTs) or state-based CRDTs.
[4]Distributed lock acquisition or leader election require an external coordination service like ZooKeeper, for example, whereas static rules do not.