Reliable Proxy Pattern
Loading

Reliable Proxy Pattern

Looking at Message Delivery Reliability one might come to the conclusion that Akka actors are made for blue-sky scenarios: sending messages is the only way for actors to communicate, and then that is not even guaranteed to work. Is the whole paradigm built on sand? Of course the answer is an emphatic “No!”.

A local message send—within the same JVM instance—is not likely to fail, and if it does the reason was one of

  • it was meant to fail (due to consciously choosing a bounded mailbox, which upon overflow will have to drop messages)
  • or it failed due to a catastrophic VM error, e.g. an OutOfMemoryError, a memory access violation (“segmentation fault”, GPF, etc.), JVM bug—or someone calling System.exit().

In all of these cases, the actor was very likely not in a position to process the message anyway, so this part of the non-guarantee is not problematic.

It is a lot more likely for an unintended message delivery failure to occur when a message send crosses JVM boundaries, i.e. an intermediate unreliable network is involved. If someone unplugs an ethernet cable, or a power failure shuts down a router, messages will be lost while the actors would be able to process them just fine.

Note

This does not mean that message send semantics are different between local and remote operations, it just means that in practice there is a difference between how good the “best effort” is.

§Introducing the Reliable Proxy

../_images/ReliableProxy.png

To bridge the disparity between “local” and “remote” sends is the goal of this pattern. When sending from A to B must be as reliable as in-JVM, regardless of the deployment, then you can interject a reliable tunnel and send through that instead. The tunnel consists of two end-points, where the ingress point P (the “proxy”) is a child of A and the egress point E is a child of P, deployed onto the same network node where B lives. Messages sent to P will be wrapped in an envelope, tagged with a sequence number and sent to E, who verifies that the received envelope has the right sequence number (the next expected one) and forwards the contained message to B. When B receives this message, the sender() will be a reference to the sender() of the original message to P. Reliability is added by E replying to orderly received messages with an ACK, so that P can tick those messages off its resend list. If ACKs do not come in a timely fashion, P will try to resend until successful.

§Exactly what does it guarantee?

Sending via a ReliableProxy makes the message send exactly as reliable as if the represented target were to live within the same JVM, provided that the remote actor system does not terminate. In effect, both ends (i.e. JVM and actor system) must be considered as one when evaluating the reliability of this communication channel. The benefit is that the network in-between is taken out of that equation.

§Connecting to the target

The proxy tries to connect to the target using the mechanism outlined in Identifying Actors via Actor Selection. Once connected, if the tunnel terminates the proxy will optionally try to reconnect to the target using using the same process.

Note that during the reconnection process there is a possibility that a message could be delivered to the target more than once. Consider the case where a message is delivered to the target and the target system crashes before the ACK is sent to the sender. After the proxy reconnects to the target it will start resending all of the messages that it has not received an ACK for, and the message that it never got an ACK for will be redelivered. Either this possibility should be considered in the design of the target or reconnection should be disabled.

§How to use it

Since this implementation does not offer much in the way of configuration, simply instantiate a proxy wrapping a target ActorPath. From Java it looks like this:

  1. import akka.contrib.pattern.ReliableProxy;
  2.  
  3.  
  4. public class ProxyParent extends UntypedActor {
  5. private final ActorRef proxy;
  6.  
  7. public ProxyParent(ActorPath targetPath) {
  8. proxy = getContext().actorOf(
  9. ReliableProxy.props(targetPath,
  10. Duration.create(100, TimeUnit.MILLISECONDS)));
  11. }
  12.  
  13. public void onReceive(Object msg) {
  14. if ("hello".equals(msg)) {
  15. proxy.tell("world!", getSelf());
  16. }
  17. }
  18. }

And from Scala like this:

  1. import akka.contrib.pattern.ReliableProxy
  2.  
  3. class ProxyParent(targetPath: ActorPath) extends Actor {
  4. val proxy = context.actorOf(ReliableProxy.props(targetPath, 100.millis))
  5.  
  6. def receive = {
  7. case "hello" proxy ! "world!"
  8. }
  9. }

Since the ReliableProxy actor is an FSM, it also offers the capability to subscribe to state transitions. If you need to know when all enqueued messages have been received by the remote end-point (and consequently been forwarded to the target), you can subscribe to the FSM notifications and observe a transition from state ReliableProxy.Active to state ReliableProxy.Idle.

  1. public class ProxyTransitionParent extends UntypedActor {
  2. private final ActorRef proxy;
  3. private ActorRef client = null;
  4.  
  5. public ProxyTransitionParent(ActorPath targetPath) {
  6. proxy = getContext().actorOf(
  7. ReliableProxy.props(targetPath,
  8. Duration.create(100, TimeUnit.MILLISECONDS)));
  9. proxy.tell(new FSM.SubscribeTransitionCallBack(getSelf()), getSelf());
  10. }
  11.  
  12. public void onReceive(Object msg) {
  13. if ("hello".equals(msg)) {
  14. proxy.tell("world!", getSelf());
  15. client = getSender();
  16. } else if (msg instanceof FSM.CurrentState<?>) {
  17. // get initial state
  18. } else if (msg instanceof FSM.Transition<?>) {
  19. @SuppressWarnings("unchecked")
  20. final FSM.Transition<ReliableProxy.State> transition =
  21. (FSM.Transition<ReliableProxy.State>) msg;
  22. assert transition.fsmRef().equals(proxy);
  23. if (transition.from().equals(ReliableProxy.active()) &&
  24. transition.to().equals(ReliableProxy.idle())) {
  25. client.tell("done", getSelf());
  26. }
  27. }
  28. }
  29. }

From Scala it would look like so:

  1. class ProxyTransitionParent(targetPath: ActorPath) extends Actor {
  2. val proxy = context.actorOf(ReliableProxy.props(targetPath, 100.millis))
  3. proxy ! FSM.SubscribeTransitionCallBack(self)
  4.  
  5. var client: ActorRef = _
  6.  
  7. def receive = {
  8. case "go"
  9. proxy ! 42
  10. client = sender()
  11. case FSM.CurrentState(`proxy`, initial)
  12. case FSM.Transition(`proxy`, from, to)
  13. if (to == ReliableProxy.Idle)
  14. client ! "done"
  15. }
  16. }

§Configuration

  • Set akka.reliable-proxy.debug to on to turn on extra debug logging for your ReliableProxy actors.
  • akka.reliable-proxy.default-connect-interval is used only if you create a ReliableProxy with no reconnections (that is, reconnectAfter == None). The default value is the value of the configuration property akka.remote.retry-gate-closed-for. For example, if akka.remote.retry-gate-closed-for is 5 s case the ReliableProxy will send an Identify message to the target every 5 seconds to try to resolve the ActorPath to an ActorRef so that messages can be sent to the target.

§The Actor Contract

§Message it Processes

  • FSM.SubscribeTransitionCallBack and FSM.UnsubscribeTransitionCallBack, see FSM
  • ReliableProxy.Unsent, see the API documentation for details.
  • any other message is transferred through the reliable tunnel and forwarded to the designated target actor

§Messages it Sends

  • FSM.CurrentState and FSM.Transition, see FSM
  • ReliableProxy.TargetChanged is sent to the FSM transition subscribers if the proxy reconnects to a new target.
  • ReliableProxy.ProxyTerminated is sent to the FSM transition subscribers if the proxy is stopped.

§Exceptions it Escalates

  • no specific exception types
  • any exception encountered by either the local or remote end-point are escalated (only fatal VM errors)

§Arguments it Takes

  • target is the ActorPath to the actor to which the tunnel shall reliably deliver messages, B in the above illustration.
  • retryAfter is the timeout for receiving ACK messages from the remote end-point; once it fires, all outstanding message sends will be retried.
  • reconnectAfter is an optional interval between connection attempts. It is also used as the interval between receiving a Terminated for the tunnel and attempting to reconnect to the target actor.
  • maxConnectAttempts is an optional maximum number of attempts to connect to the target while in the Connecting state.