测试 Actor 系统

优质
小牛编辑
141浏览
2023-12-01
  • TestKit 实例 (Scala)

对于任何软件开发,自动化测试都是开发过程中一个重要组成部分。actor 模型对于代码单元如何划分,它们之间如何交互提供了一种新的视角,这对如何编写测试也造成了影响。

Akka 有一个专门的模块——akka-testkit来支持不同层次上的测试,测试很明显有两个类别:

  • 测试独立的、不包括actor模型的代码,即没有多线程的内容;这意味着给事件发生的次序给定,有完全确定的行为,没有任何并发考虑,这在下文中称为 单元测试(Unit Testing)
  • 测试(多个)包装过的actor,包括多线程调度;这意味着事件的次序没有确定性,但由于使用了actor模型而不需要考虑并发,这在下文中被称为 集成测试(Integration Testing)

当然这两个类型有着不同的测试粒度,单元测试通常是白盒测试,而集成测试是对完整的actor网络进行的功能测试。是否把并发考虑为测试的一部分,是其中重要的区别。我们提供的工具将在下面的章节中详细介绍。

注意

请确保在依赖项中加入akka-testkit模块。

TestActorRef做同步单元测试

Actor类中的业务逻辑测试可以分为两部分:首先,每个原子操作必须独立工作,然后输入的事件序列必须被正确处理,即使事件的次序存在一些可能的变化。前者是单线程单元测试的主要使用场景,而后者可以在集成测试中进行确认。

通常,ActorRef将实际的Actor实例与外界隔离开,唯一的通信通道是actor的邮箱。这个限制是单元测试的障碍,所以我们引入了TestActorRef。这个特殊类型的引用是专门为测试而设计的,它允许以两种方式访问actor:通过获取实际actor实例的引用,或者通过调用或查询actor的行为(receive)。下文中对每一种方式都有专门的章节介绍。

获取一个Actor的引用

能够访问到实际的Actor对象使得所有的传统单元测试技术都可以用于测试其中的方法。可以像这样获取一个引用:

import akka.testkit.TestActorRef

val actorRef = TestActorRef[MyActor]
val actor = actorRef.underlyingActor

由于TestActorRef是actor类型的高阶类型(泛型),因此它返回的底层actor具有正确的静态类型。这样以后你就可以象平常一样将任何单元测试工具用于actor上了。

测试有限状态机

如果你要测试的actor是一个FSM,你可以使用专门的TestFSMRef,它拥有普通TestActorRef的所有功能,并且额外地允许访问其内部状态:

import akka.testkit.TestFSMRef
import akka.actor.FSM
import scala.concurrent.duration._

val fsm = TestFSMRef(new TestFsmActor)

val mustBeTypedProperly: TestActorRef[TestFsmActor] = fsm

assert(fsm.stateName == 1)
assert(fsm.stateData == "")
fsm ! "go" // being a TestActorRef, this runs also on the CallingThreadDispatcher
assert(fsm.stateName == 2)
assert(fsm.stateData == "go")

fsm.setState(stateName = 1)
assert(fsm.stateName == 1)

assert(fsm.isTimerActive("test") == false)
fsm.setTimer("test", 12, 10 millis, true)
assert(fsm.isTimerActive("test") == true)
fsm.cancelTimer("test")
assert(fsm.isTimerActive("test") == false)

由于Scala类型推断的限制,只有一个如上所示的工厂方法,所以你很可能需要写像TestFSMRef(new MyFSM)这样的代码,而不是想象中的类似ActorRefTestFSMRef[MyFSM]。上例所示的所有方法都直接访问FSM的状态,不作任何同步;这在使用CallingThreadDispatcher且没有其它线程参与的情况下是合适的,但如果你实际上需要处理定时器事件可能会导致意外的情形,因为它们是在Scheduler线程中执行的。

测试Actor的行为

当消息派发器调用actor的逻辑来处理一条消息时,它实际上是对当前注册到actor的行为调用了apply。行为的初始值是代码中声明的receive方法的返回值,但可以通过对外部消息的响应调用becomeunbecome来改变这个行为。所有这些特性使得进行actor的行为测试不太容易。因此TestActorRef提供了一种不同的操作方式来对Actor的测试进行补充:它支持所有正常的ActorRef中的操作。发往actor的消息在当前线程中同步处理,应答象往常一样回送。这个技巧来自下面所介绍的CallingThreadDispatcher;这个派发器被隐式地用于所有实例化为TestActorRef的actor。

import akka.testkit.TestActorRef
import scala.concurrent.duration._
import scala.concurrent.Await
import akka.pattern.ask

val actorRef = TestActorRef(new MyActor)
// hypothetical message stimulating a '42' answer
val future = actorRef ? Say42
val Success(result: Int) = future.value.get
result should be(42)

由于TestActorRefLocalActorRef的子类,只是附加了一些特殊功能,所以像监管和重启这样的功能也能正常工作,但要注意:只要所有相关的actor都使用CallingThreadDispatcher,那么所有的执行过程都是严格同步的。一旦你增加了一些元素,其中包括比较复杂的定时任务,你就离开了单元测试的范畴,因为你必须要重新将异步性纳入考虑范围(在大多数情况下问题在于要等待希望的结果有机会发生)。

另一个在单线程测试中被覆盖的特殊点是receiveTimeout,由于包含了它会产生异步的ReceiveTimeout消息队列,因此与同步约定矛盾。

警告

综上所述:TestActorRef重写了两个成员:它设置派发器为CallingThreadDispatcher.global,设置receiveTimeout为None。

介于两者之间方法:期望异常

如果你想要测试actor的行为,包括热替换,但是不涉及消息派发器,也不希望TestActorRef吞掉所有抛出的异常,那么为你准备了另一种模式:只要使用 TestActorRefreceive 方法 ,这将会把消息转发给内部的 actor:

import akka.testkit.TestActorRef

val actorRef = TestActorRef(new Actor {
  def receive = {
    case "hello" => throw new IllegalArgumentException("boom")
  }
})
intercept[IllegalArgumentException] { actorRef.receive("hello") }
使用场景

当然你可以根据自己的测试需求来混合使用 TestActorRef 的不同用法:

  • 一个常见的使用场景是在发送测试消息之前设置actor进入某个特定的内部状态
  • 另一个场景是发送了测试消息之后验证内部状态转换的正确性

放心大胆地对各种可能性进行实验,如果你发现了有用的模式,快让Akka论坛知道它!常用操作甚至可能放进优美的DSL中,谁知道呢。

TestKit进行异步集成测试

当基本确定你的actor的业务逻辑是正确的之后,下一步就是确认它在目标环境中也能正确工作(如果actor个体都足够简单,可能是因为他们使用了 FSM 模块,这也可以放到第一步)。关于环境的定义当然很大程度上由手头的问题和你打算测试的程度来决定,从功能/集成测试到完整的系统测试。最简单的步骤包括测试过程(提供所期望的触发条件)、要测试的actor和接收应答的actor。大一些的系统将被测的actor替换成一组actor网络,将触发条件应用于不同的切入点,并整理将会从不同的输出位置发送的结果,但基本的原则保持不变,就是测试由一个单独的流程来驱动。

TestKit 类包含一组工具来简化这些常用的工作.

import akka.actor.ActorSystem
import akka.actor.Actor
import akka.actor.Props
import akka.testkit.{ TestActors, TestKit, ImplicitSender }
import org.scalatest.WordSpecLike
import org.scalatest.Matchers
import org.scalatest.BeforeAndAfterAll

class MySpec(_system: ActorSystem) extends TestKit(_system) with ImplicitSender
  with WordSpecLike with Matchers with BeforeAndAfterAll {

  def this() = this(ActorSystem("MySpec"))

  override def afterAll {
    TestKit.shutdownActorSystem(system)
  }

  "An Echo actor" must {

    "send back messages unchanged" in {
      val echo = system.actorOf(TestActors.echoActorProps)
      echo ! "hello world"
      expectMsg("hello world")
    }

  }
}

TestKit 中有一个名为 testActor 的actor作为将要被不同的 expectMsg...断言检查的消息的入口,下面会详细介绍这些断言。 当混入了 ImplicitSender trait后,这个测试actor会在整个测试过程中,被隐式地用作消息的发送者引用。testActor 也可以像平常一样被发送给其它的actor,通常是订阅成为通知监听器。有一套检查方法,例如 接收所有匹配某些条件的消息,接收一系列固定的消息序列或类,在某段时间内收不到消息,等。

作为参数传递到TestKit构造函数的ActorSystem可以通过system成员来访问。记得在测试完成后关闭actor系统 (即使是在测试失败的情况下) 以保证所有的actor——包括测试actor——被停止.

内置断言

上面提到的 expectMsg并不是唯一的对收到的消息进行断言的方法。以下是完整的列表:

  • expectMsg[T](d: Duration, msg: T): T

    给定的消息对象必须在指定的时间内到达;该消息对象将被返回。

  • expectMsgPF[T](d: Duration)(pf: PartialFunction[Any, T]): T

    在给定的时间段内,必须有一条消息到达,必须为这类消息定义了偏函数;将收到的消息应用于偏函数并返回其结果。可以不指定时间段(这时需要一对空的括号),这时使用最深层的 within 块中的期限。

  • expectMsgClass[T](d: Duration, c: Class[T]): T

    在分配的时间片内必须接收到Class类型的对象实例;并返回该对象。注意它的类型匹配是子类兼容的;如果需要类型是相等的,参考使用单个class参数的expectMsgAllClassOf

  • expectMsgType[T: Manifest](d: Duration)

    在分配的时间片内必须收到指定类型 (擦除后)的对象实例;并返回该对象。这个方法基本上与 expectMsgClass(implicitly[ClassTag[T]].runtimeClass)等价。

  • expectMsgAnyOf[T](d: Duration, obj: T*): T

    在指定的时间内必须收到一个对象,而且此对象必须与传入的多个对象引用中至少一个相等( 用 == 进行比较);并返回该对象。

  • expectMsgAnyClassOf[T](d: Duration, obj: Class[_ <: T]*): T

    在指定的时间内必须收到一个对象,它必须指定的多个Class中某一个类对象的实例;并返回该对象。

  • expectMsgAllOf[T](d: Duration, obj: T*): Seq[T]

    在指定时间内必须收到与指定的对象数组中相等数量的对象,对每个收到的对象,必须至少有一个数组中的对象与之相等(用==进行比较) 。返回收到对象的完整序列。

  • expectMsgAllClassOf[T](d: Duration, c: Class[_ <: T]*): Seq[T]

    在指定时间内必须收到与指定的 Class 数组相等数量的对象,对数组中的每一个类,必须至少有一个对象的类与之相等(用 ==进行比较) (这 不是 子类兼容的类型检查)。返回收到对象的完整序列。

  • expectMsgAllConformingOf[T](d: Duration, c: Class[_ <: T]*): Seq[T]

    在指定时间内必须收到与指定的 Class数组相等数量的对象,对数组中的每个类必须至少有一个对象是这个类的实例。返回收到对象的完整序列。

  • expectNoMsg(d: Duration)

    在指定时间内不能收到消息。如果在这个方法被调用之前已经收到了消息,并且没有用其它的方法将这些消息从队列中删除,这个断言也会失败。

  • receiveN(n: Int, d: Duration): Seq[AnyRef]

    指定的时间内必须收到n条消息;返回收到的消息。

  • fishForMessage(max: Duration, hint: String)(pf: PartialFunction[Any, Boolean]): Any

    只要时间没有用完,而且偏函数匹配收到的消息并返回false,则一直接收消息。返回使偏函数返回true 的消息或抛出异常,异常中会提供一些提示供debug使用。

除了消息接收断言,还有一些方法来对消息流提供帮助:

  • receiveOne(d: Duration): AnyRef

    尝试等待给定的时间间隔来接收一个消息,如果失败则返回null。如果给定的Duration是0,则调用是非阻塞的(轮询模式)。

  • receiveWhile[T](max: Duration, idle: Duration, messages: Int)(pf: PartialFunction[Any, T]): Seq[T]

    只要满足以下条件就收集消息:

    • 消息与偏函数匹配
    • 指定的时间还没用完
    • 在空闲的时间内收到了下一条消息
    • 消息数量还没有到上限

    返回收集到的所有消息。时间上限缺省值是最深层的 within 块中的剩余时间,空闲时间缺省为无限 (也就是禁止空闲超时功能)。期望的消息数量缺省值为 Int.MaxValue,也就是不作这个限制。

  • awaitCond(p: => Boolean, max: Duration, interval: Duration)

    每经过 interval时间就检查一下给定的条件,直到它返回 true 或者 max 时间用完了。时间间隔缺省为 100 ms 而最大值缺省为最深层的 within 块中的剩余时间。

  • awaitAssert(a: => Any, max: Duration, interval: Duration)

    在每个时间间隔验证一遍给定的断言函数,直到它不抛出异常,或max持续时间用完。如果在超时已到则抛出最后一个异常。间隔默认为 100 ms 和最大值默认为最深层的 within 块中的剩余时间。 TODO Akka文档中的重复

  • ignoreMsg(pf: PartialFunction[AnyRef, Boolean]) ignoreNoMsg

    内部的 testActor 包含一个偏函数用来忽略消息:它只会将与偏函数不匹配或使函数返回false的消息放进队列。 这个函数可以用上面的方法进行设置和重设;每一次调用都会覆盖之前的函数,而不会迭加。

    这个功能在你想忽略一般的消息而只对指定的一些消息感兴趣时(例如测试日志系统时)比较有用。

期望日志消息

由于集成测试不允许进入参与测试的actor内部处理流程,无法直接确认预料中的异常。因此为了做这件事,只能使用日志系统:将普通的事件处理器替换成 TestEventListener并使用一个 EventFilter ,从而可以对日志消息,包括由于异常产生的日志,做断言:

import akka.testkit.EventFilter
import com.typesafe.config.ConfigFactory

implicit val system = ActorSystem("testsystem", ConfigFactory.parseString("""
  akka.loggers = ["akka.testkit.TestEventListener"]
  """))
try {
  val actor = system.actorOf(Props.empty)
  EventFilter[ActorKilledException](occurrences = 1) intercept {
    actor ! Kill
  }
} finally {
  shutdown(system)
}

如果出现的次数是具体的——如上所示——然后intercept将阻塞直到收到匹配消息达到这一数字,或akka.test.filter-leeway配置的超时时间用完了(在传入的代码块返回后,时间开始计数)。超时时测试失败。

注意

一定要在你的 application.conf中替换默认的日志记录器为TestEventListener,以启用此功能:

akka.loggers = [akka.testkit.TestEventListener]
对定时进行断言

功能测试的另一个重要部分与定时器有关:有些事件不能立即发生(如定时器),另外一些需要在时间期限内发生。因此所有的进行检查的方法 都接收一个时间上限,不论是正面还是负面的结果都应该在这个时间之前获得。时间下限需要在这个检测方法之外进行检查,因而有一个新的工具来管理时间期限:

   within([min, ]max) {
     ...
   }

传给within的代码块必须在一个介于 minmax之间的Duration之前完成,其中min缺省值为0。将max 参数与块的启动时间相加得到的时间期限在所有检查方法块内部都可以隐式获得,如果你没有指定max值,它会从最深层的 within 块继承这个值.

应注意如果代码块的最后一条接收消息断言是 expectNoMsgreceiveWhile,对 within 的最终检查将被跳过,以避免由于唤醒延迟导致的错误的true值。这意味着虽然其中每一个独立的断言仍然使用时间上限,整个代码块在这种情况下会有长度随机的延迟。

import akka.actor.Props
import scala.concurrent.duration._

val worker = system.actorOf(Props[Worker])
within(200 millis) {
  worker ! "some work"
  expectMsg("some result")
  expectNoMsg // will block for the rest of the 200ms
  Thread.sleep(300) // will NOT make this block fail
}

注意

所有的时间都以 System.nanoTime为单位,即它们描述的是墙上时间,而非CPU时间.

Ray Roestenburg 写了一篇关于使用 TestKit 的好文。完整的示例也可以在这里找到.

考虑很慢的测试系统

你在跑得飞快的笔记本上使用的超时设置在高负载的Jenkins(或类似的)服务器上通常都会导致虚假的测试失败。 为了考虑这种情况,所有的时间上限都在内部乘以一个系数,这个系数来自配置文件中的 akka.test.timefactor,缺省值为 1。

你也可以用akka.testkit包对象中的隐式转换来将同样的系数来作用于其它的时限,为Duration添加扩展函数。

import scala.concurrent.duration._
import akka.testkit._
10.milliseconds.dilated
用隐式的ActorRef解决冲突

如果你希望在基于TestKit的测试中,消息发送者为 testActor,只需要在你的测试代码混入 ÌmplicitSender

class MySpec(_system: ActorSystem) extends TestKit(_system) with ImplicitSender
  with WordSpecLike with Matchers with BeforeAndAfterAll {
使用多个探针 Actor

如果待测的actor会发送多个消息到不同的目标,在使用TestKit时可能会难以分辨到达 testActor的消息流。另一种方法是用它来创建简单的探针actor,将它们插入到消息流中。为了让这种方法更加强大和方便,我们提供了一个具体实现,称为 TestProbe。它的功能可以用下面的小例子说明:

import scala.concurrent.duration._
import akka.actor._
import scala.concurrent.Future

class MyDoubleEcho extends Actor {
  var dest1: ActorRef = _
  var dest2: ActorRef = _
  def receive = {
    case (d1: ActorRef, d2: ActorRef) =>
      dest1 = d1
      dest2 = d2
    case x =>
      dest1 ! x
      dest2 ! x
  }
}
val probe1 = TestProbe()
val probe2 = TestProbe()
val actor = system.actorOf(Props[MyDoubleEcho])
actor ! ((probe1.ref, probe2.ref))
actor ! "hello"
probe1.expectMsg(500 millis, "hello")
probe2.expectMsg(500 millis, "hello")

这里我们用 MyDoubleEcho来模拟一个待测系统,它会将输入镜像为两个输出。关联两个测试探针来进行(最简单)行为的确认。还有一个例子是两个相互协作的 actor A,B, A 发送消息给 B。为了确认这个消息流,可以插入一个 TestProbe 作为A的目标,使用转发功能或下文中的自动导向功能在测试上下文中包含一个真实的B。

还可以为探针配备自定义的断言来使测试代码更简洁清晰:

ase class Update(id: Int, value: String)

val probe = new TestProbe(system) {
  def expectUpdate(x: Int) = {
    expectMsgPF() {
      case Update(id, _) if id == x => true
    }
    sender() ! "ACK"
  }
}

这里你拥有完全的灵活性,可以将TestKit 提供的工具与你自己的检测代码混合和匹配,并为它取一个有意义的名字。在实际开发中你的代码很可能比上面的示例要复杂;要充分利用工具的力量!

警告

任何从 TestProbe发送到另一个运行在 CallingThreadDispatcher上的actor的消息,如果另一个actor也可能会向此探测器发送消息,则存在死锁的可能性。TestProbe.watchTestProbe.unwatch的实现也将向监控者发送消息,这意味着尝试监视它是危险的,例如从TestProbe监视TestActorRef

从探针观察其它actor

TestProbe可以将自己注册为任意其他actor的DeathWatch:

val probe = TestProbe()
probe watch target
target ! PoisonPill
probe.expectTerminated(target)
对探针收到的消息进行应答

探针在可能的条件下,会记录通讯通道以便进行应答:

val probe = TestProbe()
val future = probe.ref ? "hello"
probe.expectMsg(0 millis, "hello") // TestActor runs on CallingThreadDispatcher
probe.reply("world")
assert(future.isCompleted && future.value == Some(Success("world")))
对探针收到的消息进行转发

假定一个象征性的actor网络中某目标 actor dest 从 actor source收到一条消息。如果你安排消息使其先发往 TestProbe probe,你可以在保持网络功能的同时对消息流的容量和时限进行断言:

class Source(target: ActorRef) extends Actor {
  def receive = {
    case "start" => target ! "work"
  }
}

class Destination extends Actor {
  def receive = {
    case x => // Do something..
  }
}
val probe = TestProbe()
val source = system.actorOf(Props(classOf[Source], probe.ref))
val dest = system.actorOf(Props[Destination])
source ! "start"
probe.expectMsg("work")
probe.forward(dest)

目标dest actor 将收到同样的消息,就象没有插入探针一样。

自动导向

将收到的消息放进队列以便以后处理,这种方法不错,但要保持测试运行并对其运行过程进行跟踪,你也可以为参与测试的探针(事实上是任何 TestKit)安装一个 AutoPilot(自动导向)。自动导向在消息进入检查队列之前启动。以下代码可以用来转发消息,例如 A --> Probe --> B,只要满足一定的协议。

val probe = TestProbe()
probe.setAutoPilot(new TestActor.AutoPilot {
  def run(sender: ActorRef, msg: Any): Option[TestActor.AutoPilot] =
    msg match {
      case "stop" ⇒ None
      case x      ⇒ testActor.tell(x, sender); Some(this)
    }
})

run 方法必须返回auto-pilot供下一条消息使用,它可以是KeepRunning来保存当前值,或者是NoAutoPilot来终止自动导向。

小心定时器断言

在使用测试探针时,within块的行为可能会不那么直观:你需要记住上文 所描述的nicely scoped期限仅对每一个探针的局部作用域有效。因此,探针不会响应别的探针的期限,也不响应包含它的TestKit实例的期限:

val probe = TestProbe()
within(1 second) {
  probe.expectMsg("hello")
}

这里expectMsg调用将使用默认超时。

CallingThreadDispatcher

如上文所述,CallingThreadDispatcher在单元测试中非常重要,但最初它出现是为了在出错的时候能够生成连续的stacktrace。由于这个特殊的派发器一般地将任何消息直接运行在当前线程中,所以只要所有的actor都是在这个派发器上运行,消息处理的完整历史信息在调用堆栈上就有记录。

如何使用它

只要象平常一样设置派发器:

import akka.testkit.CallingThreadDispatcher
val ref = system.actorOf(Props[MyActor].withDispatcher(CallingThreadDispatcher.Id))
它是如何运作的

在被调用时,CallingThreadDispatcher 会检查接收消息的actor是否已经在当前线程中了。这种情况的最简单的例子是actor向自己发送消息。这时,不能马上对它进行处理,因为这违背了actor模型,于是这个消息被放进队列,直到actor的当前消息被处理完毕;因此,新消息会在调用的线程上被处理,只是在actor完成其先前的工作之后。在别的情况下,消息会在当前线程中立即得到处理。通过这个派发器规划的Future也会立即执行。

这种工作方式使 CallingThreadDispatcher 象一个为永远不会因为外部事件而阻塞的actor所设计的通用派发器。

在有多个线程的情况下,有可能同时存在两个使用这个派发器的actor在不同线程中收到消息。此时,它们会立即在自己的线程中被执行,并竞争actor锁,竞争失败的那个必须等待。 这样我们保持了actor模型,但由于使用了受限的调度我们损失了一些并发性。从这个意义上说,它等同于使用传统的基于互斥的并发。

另一个困难是正确地处理挂起和继续:当actor被挂起时,后续的消息将被放进一个thread-local的队列中(和正常情况下使用的队列是同一个)。但是对 resume的调用,是由一个特定的线程执行的,系统中所有其它的线程很可能不会运行这个特定的actor,这会导致thread-local队列无法被它们的本地线程清空。于是,调用 resume 的线程会从所有线程收集所有当前在队列中的消息到自己的队列中,然后处理它们。

局限性

警告

CallingThreadDispatcher被用作顶级actor,但没有通过TestActorRef的情况下,则存在一个时间窗,在此期间actor会等待user 守护者actor的构造。在此期间发送到这个actor的消息会被加入队列,然后在守护者的线程,而不是调用者的线程上执行。要避免此问题,请使用 TestActorRef

如果一个actor发送完消息后由于某种原因(通常是被发完消息后的调用actor所影响)阻塞了,此时若使用这个派发器,显然将导致死锁。 这在使用基于 CountDownLatch 同步actor测试中是很常见的情景:

   val latch = new CountDownLatch(1)
   actor ! startWorkAfter(latch)   // actor will call latch.await() before proceeding
   doSomeSetupStuff()
   latch.countDown()

这个例子将无限挂起,消息处理到达第二行而永远到不了第四行,而只有在第四行才能在一个普通的派发器上取消它的阻塞。

所以要记住 CallingThreadDispatcher并不是普通派发器的通用替代品。而另一方面在它上面运行你的actor网络测试会非常有用,因为如果它在机率特别高的条件下都能不死锁,那么在生产环境中也不会。

警告

上面这句话很遗憾并不是一个有力的保证,因为你的代码运行在不同的派发器上时可能直接或间接地改变它的行为。 如果你想要寻找帮助你debug死锁的工具,CallingThreadDispatcher 在有些错误场合下可能会有用,但要记住它既可能给出错误的正面结果也可能给出错误的负面结果。

线程中断

如果 CallingThreadDispatcher看到当前线程在消息处理返回时已设置其 isInterrupted() 标志,它将在完成所有其处理(即,如上文所述的需要处理的所有消息,都会在这种情况发生之前处理)后抛出InterruptedException异常。正如tell由于其契约不能抛出异常,该异常将然后被捕获并记录日志,并且在该线程中断状态将再次设置。

如果在消息处理过程中抛出InterruptedException异常,则它将被CallingThreadDispatcher 的消息处理循环捕获,然后将设置该线程interrupted标志,并且处理将继续正常进行。

注意

这两个段落的总结是,如果当前线程在CallingThreadDispatcher下工作时中断,则将导致当消息发送返回时 isInterrupted标志被置为true,并不抛出InterruptedException异常。

好处

综上所述,以下是 CallingThreadDispatcher能够提供的特性:

  • 确定地执行单线程测试,同时保持几乎所有的actor语义
  • 在异常stacktrace中记录从失败点开始的完整的消息处理历史
  • 排除某些类型的死锁场景

跟踪Actor调用

到目前为止所有的测试工具都针对系统的行为构造断言。如果测试失败,通常是由你来查找原因,进行修改并进行下一轮验证测试。这个过程既有debugger支持,又有日志支持,这里Akka工具箱提供以下选项:

  • 对Actor实例中抛出的异常记录日志

    相比其它的日志机制,这一条是永远打开的;它的日志级别是 ERROR.

  • 对某些actor的消息调用记录日志

    这是通过在配置文件里添加设置项来打开 — 即 akka.actor.debug.receive — 它使得 loggable语句被应用在actor的 receive 函数上:

import akka.event.LoggingReceive
def receive = LoggingReceive {
  case msg => // Do something ...
}
def otherState: Receive = LoggingReceive.withLabel("other") {
  case msg => // Do something else ...
}

如果在配置文件中没有上面给出的配置,这个方法将直接移交给给定的未被修改的 Receive 函数,也就是说如果不打开,就没有运行时开销。

这个日志功能是与指定的局部标记绑定的,因为将其一致地应用于所有的actor一般不是你所需要的,如果被用于事件主线日志监听器,它还可能导致无限循环。

  • 对特殊的消息记录日志

Actor会自动处理某些特殊消息,例如KillPoisonPill 等等。打开对这些消息的跟踪只需要设置 akka.actor.debug.autoreceive,这对所有actor都有效。

  • 对actor生命周期记录日志

Actor的创建、启动、重启、开始监控、停止监控和终止可以通过打开 akka.actor.debug.lifecycle 来跟踪;这也是对所有actor都有效的。

所有这些日志消息都记录在 DEBUG 级别。总结一下,你可以用以下配置片段打开actor活动的完整日志:

akka {
  loglevel = DEBUG
  actor {
    debug {
      receive = on
      autoreceive = on
      lifecycle = on
    }
  }
}

不同的测试框架

Akka的测试套件是使用ScalaTest,并参照文档中的示例编写的。然而,TestKit和其工具包并不依赖于这一框架,实际上你可以使用最适合你开发风格的测试框架。

本节包含使用其他框架时的已知陷阱,这不是详尽的,并不意味着需要认证或使用特殊支持。

当你需要它是一个特质

如果由于某种原因是,TestKit是一个具体的类,而不是特质,因此测试工具无法继承它,还有TestKitBase

import akka.testkit.TestKitBase

class MyTest extends TestKitBase {
  implicit lazy val system = ActorSystem()

  // put your test code here ...

  shutdown(system)
}

implicit lazy val system必须这样声明 (你当然可以根据需要将参数传递到actor系统工厂中) 因为特质 TestKitBase 在其构造过程中需要system。

警告

不提倡使用这个特质,由于在未来二进制向后兼容性可能会成为问题,所以使用它须自行承担风险。

Specs2

一些 Specs2用户贡献了如何变通解决可能出现的一些冲突的示例:

  • 在TestKit中混入 org.specs2.mutable.Specification 将导致涉及 end 方法的名称冲突 (它在TestKit中是私有变量,而在Specification中是抽象方法);如果第一次混入测试工具包,代码可以编译,但可能在运行时失败。变通方法——实际上也有利于第三点—— 是将org.specs2.specification.Scope和TestKit一起应用。

  • Specification特质提供了一个Duration DSL 的部分,使用了与scala.concurrent.duration.Duration具有相同名称的方法,如果引入scala.concurrent.duration._会导致含糊不清的隐式值。有两个变通:

    • 要么使用Specification的Duration变体,提供到Akka Duration的隐式转换。这种转换不由Akka提供,因为这将意味着我们的 JAR 文件将依赖于 Specs2,为实现这样一个小特性不合理。
    • 或混入org.specs2.time.NoTimeConversions到Specification中。
  • Specifications默认是并发执行的,因此执行写入测试或者sequential关键字时需要谨慎一些。

配置

TestKit模块的几个配置属性,请参阅配置参考。