1

我正在编写一个UCI解释器,作为 Akka 有限状态机。根据规范,解释器必须将其输出写入stdout,并从stdin.

我有一个演员的测试服,我可以测试其行为的某些方面(与消息相关),但我不知道如何捕获stdout做出断言,也不知道如何通过stdin. 我已经尽我所能探索了最大规模的 API,但找不到如何实现我所需要的。

这是当前的测试类:

package org.chess

import akka.actor.ActorSystem
import akka.testkit.{TestKit, TestProbe}
import org.chess.Keyword.Quit
import org.scalatest.wordspec.AnyWordSpecLike
import org.scalatest.{BeforeAndAfterAll, Matchers}

import scala.concurrent.duration._
import scala.language.postfixOps

class UCIInterpreterSpec(_system: ActorSystem)
  extends TestKit(_system)
    with Matchers
    with AnyWordSpecLike
    with BeforeAndAfterAll {

  def this() = this(ActorSystem("UCIInterpreterSpec"))

  override def afterAll: Unit = {
    super.afterAll()
    shutdown(system)
  }

  "A UCI interpreter" should {

    "be able to quit" in {
      val testProbe = TestProbe()
      val interpreter = system.actorOf(UCIInterpreter.props)
      testProbe watch interpreter
      interpreter ! Command(Quit, Nil)
      testProbe.expectTerminated(interpreter, 3 seconds)
    }

  }

}

当然,知道解释器可以退出是有用的……但不是很有用。我需要测试,例如,如果将字符串发送isready给解释器,它会返回readyok.

我是否有可能通过使用 akka.testkit 而不是更简单的框架使测试过于复杂?为了简单起见,我想继续使用单一的测试框架,并且我需要测试系统的许多其他与参与者相关的元素,所以如果可以在不离开 akka-testkit/scalatest 域的情况下解决这个问题,那就太棒了。

任何帮助将不胜感激。提前致谢。

4

1 回答 1

4

您需要更改 Actor 的设计。

Actor 不应直接读取stdin或写入stdout。相反,在Props提供输入和接受输出的参与者对象中给出。每次需要输入时都可以调用stdin类似的方法。可能是每次生成输出时调用。或者,您可以使用或类似的构造,这些构造被设计为抽象的数据源和接收器。() => StringstdoutString => UnitStreams

在生产代码中,您传递使用stdinand的对象stdout,但对于测试代码,您传递读取和写入内存缓冲区的对象。然后,您可以检查 Actor 是否使用了适当的输入,以及 Actor 是否生成了适当的输出。

于 2019-07-02T22:05:11.020 回答