|
Akka/Scala example source code file (cluster-singleton.rst)
The cluster-singleton.rst Akka example source code.. _cluster-singleton: Cluster Singleton ================= For some use cases it is convenient and sometimes also mandatory to ensure that you have exactly one actor of a certain type running somewhere in the cluster. Some examples: * single point of responsibility for certain cluster-wide consistent decisions, or coordination of actions across the cluster system * single entry point to an external system * single master, many workers * centralized naming service, or routing logic Using a singleton should not be the first design choice. It has several drawbacks, such as single-point of bottleneck. Single-point of failure is also a relevant concern, but for some cases this feature takes care of that by making sure that another singleton instance will eventually be started. The cluster singleton pattern is implemented by ``akka.contrib.pattern.ClusterSingletonManager``. It manages one singleton actor instance among all cluster nodes or a group of nodes tagged with a specific role. ``ClusterSingletonManager`` is an actor that is supposed to be started on all nodes, or all nodes with specified role, in the cluster. The actual singleton actor is started by the ``ClusterSingletonManager`` on the oldest node by creating a child actor from supplied ``Props``. ``ClusterSingletonManager`` makes sure that at most one singleton instance is running at any point in time. The singleton actor is always running on the oldest member, which can be determined by ``Member#isOlderThan``. This can change when removing that member from the cluster. Be aware that there is a short time period when there is no active singleton during the hand-over process. The cluster failure detector will notice when oldest node becomes unreachable due to things like JVM crash, hard shut down, or network failure. Then a new oldest node will take over and a new singleton actor is created. For these failure scenarios there will not be a graceful hand-over, but more than one active singletons is prevented by all reasonable means. Some corner cases are eventually resolved by configurable timeouts. You can access the singleton actor by using the provided ``akka.contrib.pattern.ClusterSingletonProxy``, which will route all messages to the current instance of the singleton. The proxy will keep track of the oldest node in the cluster and resolve the singleton's ``ActorRef`` by explicitly sending the singleton's ``actorSelection`` the ``akka.actor.Identify`` message and waiting for it to reply. This is performed periodically if the singleton doesn't reply within a certain (configurable) time. Given the implementation, there might be periods of time during which the ``ActorRef`` is unavailable, e.g., when a node leaves the cluster. In these cases, the proxy will stash away all messages until it is able to identify the singleton. It's worth noting that messages can always be lost because of the distributed nature of these actors. As always, additional logic should be implemented in the singleton (acknowledgement) and in the client (retry) actors to ensure at-least-once message delivery. Potential problems to be aware of --------------------------------- This pattern may seem to be very tempting to use at first, but it has several drawbacks, some of them are listed below: * the cluster singleton may quickly become a *performance bottleneck*, * you can not rely on the cluster singleton to be *non-stop* available — e.g. when the node on which the singleton has been running dies, it will take a few seconds for this to be noticed and the singleton be migrated to another node, * in the case of a *network partition* appearing in a Cluster that is using Automatic Downing (see Auto Downing docs for :ref:`Scala <automatic-vs-manual-downing-scala>` or :ref:`Java <automatic-vs-manual-downing-java>`), it may happen that the isolated clusters each decide to spin up their own singleton, meaning that there might be multiple singletons running in the system, yet the Clusters have no way of finding out about them (because of the partition). Especially the last point is something you should be aware of — in general when using the Cluster Singleton pattern you should take care of downing nodes yourself and not rely on the timing based auto-down feature. .. warning:: **Be very careful when using Cluster Singleton together with Automatic Downing**, since it allows the cluster to split up into two separate clusters, which in turn will result in *multiple Singletons* being started, one in each separate cluster! An Example ---------- Assume that we need one single entry point to an external system. An actor that receives messages from a JMS queue with the strict requirement that only one JMS consumer must exist to be make sure that the messages are processed in order. That is perhaps not how one would like to design things, but a typical real-world scenario when integrating with external systems. On each node in the cluster you need to start the ``ClusterSingletonManager`` and supply the ``Props`` of the singleton actor, in this case the JMS queue consumer. In Scala: .. includecode:: @contribSrc@/src/multi-jvm/scala/akka/contrib/pattern/ClusterSingletonManagerSpec.scala#create-singleton-manager Here we limit the singleton to nodes tagged with the ``"worker"`` role, but all nodes, independent of role, can be used by specifying ``None`` as ``role`` parameter. The corresponding Java API for the ``singeltonProps`` function is ``akka.contrib.pattern.ClusterSingletonPropsFactory``. The Java API takes a plain String for the role parameter and ``null`` means that all nodes, independent of role, are used. In Java: .. includecode:: @contribSrc@/src/test/java/akka/contrib/pattern/ClusterSingletonManagerTest.java#create-singleton-manager .. note:: The ``singletonProps``/``singletonPropsFactory`` is invoked when creating the singleton actor and it must not use members that are not thread safe, e.g. mutable state in enclosing actor. Here we use an application specific ``terminationMessage`` to be able to close the resources before actually stopping the singleton actor. Note that ``PoisonPill`` is a perfectly fine ``terminationMessage`` if you only need to stop the actor. Here is how the singleton actor handles the ``terminationMessage`` in this example. .. includecode:: @contribSrc@/src/multi-jvm/scala/akka/contrib/pattern/ClusterSingletonManagerSpec.scala#consumer-end Note that you can send back current state to the ``ClusterSingletonManager`` before terminating. This message will be sent over to the ``ClusterSingletonManager`` at the new oldest node and it will be passed to the ``singletonProps`` factory when creating the new singleton instance. With the names given above, access to the singleton can be obtained from any cluster node using a properly configured proxy. In Scala: .. includecode:: @contribSrc@/src/multi-jvm/scala/akka/contrib/pattern/ClusterSingletonManagerSpec.scala#create-singleton-proxy In Java: .. includecode:: @contribSrc@/src/test/java/akka/contrib/pattern/ClusterSingletonManagerTest.java#create-singleton-proxy A more comprehensive sample is available in the `Typesafe Activator <http://www.typesafe.com/platform/getstarted>`_ tutorial named `Distributed workers with Akka and Scala! <http://www.typesafe.com/activator/template/akka-distributed-workers>`_ and `Distributed workers with Akka and Java! <http://www.typesafe.com/activator/template/akka-distributed-workers-java>`_. Other Akka source code examplesHere is a short list of links related to this Akka cluster-singleton.rst source code file: |
... this post is sponsored by my books ... | |
#1 New Release! |
FP Best Seller |
Copyright 1998-2024 Alvin Alexander, alvinalexander.com
All Rights Reserved.
A percentage of advertising revenue from
pages under the /java/jwarehouse
URI on this website is
paid back to open source projects.