All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.teambytes.inflatable.raft.RaftClientActor.scala Maven / Gradle / Ivy

The newest version!
package com.teambytes.inflatable.raft

import akka.event.Logging.simpleName
import akka.actor._
import akka.pattern.ask
import protocol._
import concurrent.duration._
import scala.concurrent.forkjoin.ThreadLocalRandom
import scala.collection.immutable.{TreeSet, Set}
import com.teambytes.inflatable.raft.config.RaftConfiguration
import akka.event.Logging
import scala.concurrent.Future
import akka.util.Timeout

/**
 * This actor hides the complexity of being a raft client from you (for example the fact of having to talk with the Leader,
 * when performing writes is abstracted away using this actor).
 *
 *
 * TODO
 * https://github.com/ktoso/akka-raft/issues/13
 * https://github.com/ktoso/akka-raft/issues/15
 *
 * Note to self, I think a proxy one will be easier to implement, because we can delay sending msgs, until we get info
 * that a leader was selected, and it's easier to retry sending hm hm... We'll see.
 *
 * @param raftMembersPaths actor path used to select raft cluster members (suffix it with `*` to glob for many actors)
 */
private[inflatable] class RaftClientActor(raftMembersPaths: ActorPath*) extends Actor with ActorLogging with Stash {

  val settings = RaftConfiguration(context.system)
  val debug = settings.publishTestingEvents

  implicit val clientDispatcher = context.system.dispatchers.lookup("raft-client-dispatcher")

  protected var members = TreeSet.empty[ActorRef]
  protected var leader: Option[ActorRef] = None

  override def preStart() {
    super.preStart()

    self ! FindLeader(delay = 1000.milliseconds)
  }

  def receive = {

    // leader handling
    case LeaderIs(Some(newLeader), maybeMsg) =>
      log.info("Member {} informed RaftClient that Leader is {} (previously assumed: {})", sender(), newLeader, leader)
      leader = Some(newLeader)
      unstashAll()

    case LeaderIs(None, maybeMsg) =>
      log.info("Member {} thinks there is no Leader currently in the raft cluster", sender())
//      self ! FindLeader


    // leader finding
    case FindLeader(delay) =>
      asyncRefreshMembers()
      randomMember() foreach { _ ! WhoIsTheLeader }

      if (leader.isEmpty)
        context.system.scheduler.scheduleOnce(delay, self, FindLeader(delay))

    case ActorIdentity(_, member) =>
//      log.info("Adding member {} to known members set: {}", member, members)
      member foreach { members += _ }

    // message handling
    case null =>
      log.warning("null sent as message to {}, ignoring!", simpleName(this))
      // ignore...

    case wrapped: ClientMessage[_] =>
      proxyOrStash(wrapped.cmd, wrapped.client)

    case msg =>
      proxyOrStash(msg, sender())
  }


  /** @param userActor end-user actor whom for we're proxying this command */
  def proxyOrStash(msg: Any, userActor: AnyRef) {
    log.info("got : " + msg)
    leader match {
      case Some(lead) =>
        if (debug) publishDebug(s"Proxying ${ClientMessage(sender(), msg)}")
        log.info("Proxying message {} from {} to currently assumed leader {}.", msg, sender(), lead)
        lead forward ClientMessage(sender(), msg)

      case _ =>
        if (debug) publishDebug("Stashing $msg, because no leader known")
        log.info("No leader in cluster, stashing client message of type {} from {}. Will re-send once leader elected.", msg.getClass, sender())
        stash()
    }
  }

  private def asyncRefreshMembers() {
    raftMembersPaths foreach { actorPath =>
      log.info("Selecting on {}", actorPath)
      context.system.actorSelection(actorPath) ! Identify()
    }
  }

  @inline private def publishDebug(msg: String) {
    context.system.eventStream.publish(Logging.Debug(simpleName(this), getClass, msg))
  }

  private def randomMember(): Option[ActorRef] =
    if (members.isEmpty) None
    else members.drop(ThreadLocalRandom.current nextInt members.size).headOption

  final case class FindLeader(delay: FiniteDuration) extends Message[Internal]
}

private[inflatable] object RaftClientActor {
  def props(raftMembersPaths: ActorPath*) =
    Props(classOf[RaftClientActor], raftMembersPaths)
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy