在maxNrOfRetries之后在SupervisorStrategy上应用自定义操作?

时间:2015-07-24 19:02:39

标签: scala akka typesafe-stack

我的家长演员看起来像

case object StartRemoteProcessor

class ConnectorActor extends Actor with ActorLogging {
  override def supervisorStrategy = OneForOneStrategy(maxNrOfRetries = 3, withinTimeRange = 20 seconds) {
    case e: OutOfMemoryError =>
      log.error("Exception received => " + e.getMessage)
      Restart
    case e: IllegalArgumentException =>
      log.error("Exception received => " + e.getMessage)
      Restart
  }

  def receive = LoggingReceive {
    case StartRemoteProcessor =>
      val remoteProcessor = context.actorOf(Props[ProcessingActor], "processingActor")
      log.info("Starting Remote Processor")
      remoteProcessor ! "Start"
    case "ProcessingStopped" =>
      notifyFailure()
  }

  def notifyFailure() = {
    log.info("notifying failure to server")
  }
}

根据docs

  

如果超出限制,则会停止子actor。

要求

  • 一旦maxNrOfRetries用尽,我想在演员最终停止时采取自定义动作notifyFailure。这将发送电子邮件

在我的孩子Actor我有

class ProcessingActor extends Actor with ActorLogging {

  override def aroundPostRestart(reason: Throwable): Unit = self.tell("Start", context.parent)
  override def preStart(): Unit = ()
  override def postStop(): Unit = context.parent ! "ProcessingStopped"

  def receive = LoggingReceive {
    case "Start" =>
      log.info("ProcessingActor path => " + self.path)
      startProcessing()
  }

  def startProcessing() = {
    println("executing startProcessing")
    throw new IllegalArgumentException("not implemented by choice")
  }
}

但是在日志中,我看到每个notifyFailure

都会调用Restart
[INFO] [07/24/2015 11:57:50.107] [main] [Remoting] Starting remoting
[INFO] [07/24/2015 11:57:50.265] [main] [Remoting] Remoting started; listening on addresses :[akka.tcp://ConnectorSystem@127.0.0.1:2554]
[INFO] [07/24/2015 11:57:50.266] [main] [Remoting] Remoting now listens on addresses: [akka.tcp://ConnectorSystem@127.0.0.1:2554]
ConnectorSystem Started
[INFO] [07/24/2015 11:57:50.277] [ConnectorSystem-akka.actor.default-dispatcher-3] [akka.tcp://ConnectorSystem@127.0.0.1:2554/user/connectorActor] Starting Remote Processor
[ERROR] [07/24/2015 11:57:50.509] [ConnectorSystem-akka.actor.default-dispatcher-4] [akka.tcp://ConnectorSystem@127.0.0.1:2554/user/connectorActor] Exception received => not implemented by choice
[ERROR] [07/24/2015 11:57:50.511] [ConnectorSystem-akka.actor.default-dispatcher-4] [akka.tcp://ProcessingSystem@127.0.0.1:2552/remote/akka.tcp/ConnectorSystem@127.0.0.1:2554/user/connectorActor/processingActor] not implemented by choice
java.lang.IllegalArgumentException: not implemented by choice
    at com.learn.remote.processing.ProcessingActor.startProcessing(ProcessingActor.scala:23)
    at com.learn.remote.processing.ProcessingActor$$anonfun$receive$1.applyOrElse(ProcessingActor.scala:18)
    at akka.actor.Actor$class.aroundReceive(Actor.scala:467)
    at com.learn.remote.processing.ProcessingActor.aroundReceive(ProcessingActor.scala:7)
    at akka.actor.ActorCell.receiveMessage(ActorCell.scala:516)
    at akka.actor.ActorCell.invoke(ActorCell.scala:487)
    at akka.dispatch.Mailbox.processMailbox(Mailbox.scala:238)
    at akka.dispatch.Mailbox.run(Mailbox.scala:220)
    at akka.dispatch.ForkJoinExecutorConfigurator$AkkaForkJoinTask.exec(AbstractDispatcher.scala:397)
    at scala.concurrent.forkjoin.ForkJoinTask.doExec(ForkJoinTask.java:260)
    at scala.concurrent.forkjoin.ForkJoinPool$WorkQueue.runTask(ForkJoinPool.java:1339)
    at scala.concurrent.forkjoin.ForkJoinPool.runWorker(ForkJoinPool.java:1979)
    at scala.concurrent.forkjoin.ForkJoinWorkerThread.run(ForkJoinWorkerThread.java:107)

[INFO] [07/24/2015 11:57:50.526] [ConnectorSystem-akka.actor.default-dispatcher-4] [akka.tcp://ConnectorSystem@127.0.0.1:2554/user/connectorActor] notifying failure to server
[ERROR] [07/24/2015 11:57:50.528] [ConnectorSystem-akka.actor.default-dispatcher-4] [akka.tcp://ConnectorSystem@127.0.0.1:2554/user/connectorActor] Exception received => not implemented by choice
[ERROR] [07/24/2015 11:57:50.528] [ConnectorSystem-akka.actor.default-dispatcher-4] [akka.tcp://ProcessingSystem@127.0.0.1:2552/remote/akka.tcp/ConnectorSystem@127.0.0.1:2554/user/connectorActor/processingActor] not implemented by choice
java.lang.IllegalArgumentException: not implemented by choice
    at com.learn.remote.processing.ProcessingActor.startProcessing(ProcessingActor.scala:23)
    at com.learn.remote.processing.ProcessingActor$$anonfun$receive$1.applyOrElse(ProcessingActor.scala:18)
    at akka.actor.Actor$class.aroundReceive(Actor.scala:467)
    at com.learn.remote.processing.ProcessingActor.aroundReceive(ProcessingActor.scala:7)
    at akka.actor.ActorCell.receiveMessage(ActorCell.scala:516)
    at akka.actor.ActorCell.invoke(ActorCell.scala:487)
    at akka.dispatch.Mailbox.processMailbox(Mailbox.scala:238)
    at akka.dispatch.Mailbox.run(Mailbox.scala:220)
    at akka.dispatch.ForkJoinExecutorConfigurator$AkkaForkJoinTask.exec(AbstractDispatcher.scala:397)
    at scala.concurrent.forkjoin.ForkJoinTask.doExec(ForkJoinTask.java:260)
    at scala.concurrent.forkjoin.ForkJoinPool$WorkQueue.runTask(ForkJoinPool.java:1339)
    at scala.concurrent.forkjoin.ForkJoinPool.runWorker(ForkJoinPool.java:1979)
    at scala.concurrent.forkjoin.ForkJoinWorkerThread.run(ForkJoinWorkerThread.java:107)

[INFO] [07/24/2015 11:57:50.533] [ConnectorSystem-akka.actor.default-dispatcher-3] [akka.tcp://ConnectorSystem@127.0.0.1:2554/user/connectorActor] notifying failure to server
[ERROR] [07/24/2015 11:57:50.534] [ConnectorSystem-akka.actor.default-dispatcher-4] [akka.tcp://ConnectorSystem@127.0.0.1:2554/user/connectorActor] Exception received => not implemented by choice
[ERROR] [07/24/2015 11:57:50.534] [ConnectorSystem-akka.actor.default-dispatcher-4] [akka.tcp://ProcessingSystem@127.0.0.1:2552/remote/akka.tcp/ConnectorSystem@127.0.0.1:2554/user/connectorActor/processingActor] not implemented by choice
java.lang.IllegalArgumentException: not implemented by choice
    at com.learn.remote.processing.ProcessingActor.startProcessing(ProcessingActor.scala:23)
    at com.learn.remote.processing.ProcessingActor$$anonfun$receive$1.applyOrElse(ProcessingActor.scala:18)
    at akka.actor.Actor$class.aroundReceive(Actor.scala:467)
    at com.learn.remote.processing.ProcessingActor.aroundReceive(ProcessingActor.scala:7)
    at akka.actor.ActorCell.receiveMessage(ActorCell.scala:516)
    at akka.actor.ActorCell.invoke(ActorCell.scala:487)
    at akka.dispatch.Mailbox.processMailbox(Mailbox.scala:238)
    at akka.dispatch.Mailbox.run(Mailbox.scala:220)
    at akka.dispatch.ForkJoinExecutorConfigurator$AkkaForkJoinTask.exec(AbstractDispatcher.scala:397)
    at scala.concurrent.forkjoin.ForkJoinTask.doExec(ForkJoinTask.java:260)
    at scala.concurrent.forkjoin.ForkJoinPool$WorkQueue.runTask(ForkJoinPool.java:1339)
    at scala.concurrent.forkjoin.ForkJoinPool.runWorker(ForkJoinPool.java:1979)
    at scala.concurrent.forkjoin.ForkJoinWorkerThread.run(ForkJoinWorkerThread.java:107)

[INFO] [07/24/2015 11:57:50.538] [ConnectorSystem-akka.actor.default-dispatcher-3] [akka.tcp://ConnectorSystem@127.0.0.1:2554/user/connectorActor] notifying failure to server
[ERROR] [07/24/2015 11:57:50.540] [ConnectorSystem-akka.actor.default-dispatcher-4] [akka.tcp://ConnectorSystem@127.0.0.1:2554/user/connectorActor] Exception received => not implemented by choice
[ERROR] [07/24/2015 11:57:50.540] [ConnectorSystem-akka.actor.default-dispatcher-4] [akka.tcp://ProcessingSystem@127.0.0.1:2552/remote/akka.tcp/ConnectorSystem@127.0.0.1:2554/user/connectorActor/processingActor] not implemented by choice
java.lang.IllegalArgumentException: not implemented by choice
    at com.learn.remote.processing.ProcessingActor.startProcessing(ProcessingActor.scala:23)
    at com.learn.remote.processing.ProcessingActor$$anonfun$receive$1.applyOrElse(ProcessingActor.scala:18)
    at akka.actor.Actor$class.aroundReceive(Actor.scala:467)
    at com.learn.remote.processing.ProcessingActor.aroundReceive(ProcessingActor.scala:7)
    at akka.actor.ActorCell.receiveMessage(ActorCell.scala:516)
    at akka.actor.ActorCell.invoke(ActorCell.scala:487)
    at akka.dispatch.Mailbox.processMailbox(Mailbox.scala:238)
    at akka.dispatch.Mailbox.run(Mailbox.scala:220)
    at akka.dispatch.ForkJoinExecutorConfigurator$AkkaForkJoinTask.exec(AbstractDispatcher.scala:397)
    at scala.concurrent.forkjoin.ForkJoinTask.doExec(ForkJoinTask.java:260)
    at scala.concurrent.forkjoin.ForkJoinPool$WorkQueue.runTask(ForkJoinPool.java:1339)
    at scala.concurrent.forkjoin.ForkJoinPool.runWorker(ForkJoinPool.java:1979)
    at scala.concurrent.forkjoin.ForkJoinWorkerThread.run(ForkJoinWorkerThread.java:107)

[INFO] [07/24/2015 11:57:50.545] [ConnectorSystem-akka.actor.default-dispatcher-4] [akka.tcp://ConnectorSystem@127.0.0.1:2554/user/connectorActor] notifying failure to server

我该如何实现这种行为?

2 个答案:

答案 0 :(得分:1)

白日梦的回答是可行的,但另一种方法可能是从父母那里观看儿童演员,当你收到被终止的消息时,执行notifyFailure

var remoteProcessor:ActorRef = _

def receive = LoggingReceive {
  case StartRemoteProcessor =>
    remoteProcessor = context.actorOf(Props[ProcessingActor], "processingActor")
    context.watch(remoteProcessor)
    log.info("Starting Remote Processor")
    remoteProcessor ! "Start"

  case Terminated(remoteProcessor) =>
    notifyFailure()
}

这样您就不需要自定义Actor生命周期方法,我发现这些方法可能是一个丰富的bug源。

答案 1 :(得分:0)

根据@rkuhn对gitter聊天的建议,以下内容适用于我

override def preRestart(reason: Throwable, message: Option[Any]): Unit = ()

所有代码?

  override def aroundPostRestart(reason: Throwable): Unit = self.tell("Start", context.parent)
  override def preRestart(reason: Throwable, message: Option[Any]): Unit = ()
  override def postStop(): Unit = context.parent ! "ProcessingStopped"