1

我有戏!2 应用程序(Scala),我有一些类在遇到错误时需要执行日志记录。

我希望能够对这些日志记录操作实际发生在正确的条件下进行单元测试。为此,我需要能够模拟记录器,但我遇到了 Mockito 的一些问题。我所拥有的(简化的)看起来像

import play.api.{ Logger, LoggerLike }

trait BaseService {
  val log: LoggerLike

  def fail(reason: String) {
    log.error(reason)
   }
}

object Service extends BaseService {
  val log = Logger
}

然后在测试中

import org.specs2.mutable._
import org.specs2.mock._
import services.BaseService

object Service extends BaseService with Mockito {
  val log = mock[play.api.Logger]

  def verify(key: String) = {
    there was one(log).error(key)
  }
}

class ServiceSpec extends Specification {
  "failures should be logged" in {
    Service.fail("foo")
    Service.verify("foo")
  }
}

但我得到了错误

[error]     NullPointerException: null (Logger.scala:43)
[error] play.api.LoggerLike$class.isErrorEnabled(Logger.scala:43)
[error] play.api.Logger.isErrorEnabled(Logger.scala:147)
[error] play.api.LoggerLike$class.error(Logger.scala:127)
[error] play.api.Logger.error(Logger.scala:147)
[error] services.BaseService$class.fail(Service.scala:19)
[error] Service$.fail(ServiceSpec.scala:11)
...

我尝试添加

log.isErrorEnabled returns true
log.error(any[String]) returns {}

但随后即使服务的初始化也失败了。

4

1 回答 1

1

我不是 Mockito 专家,但我认为您的模拟在这里出现故障,因为堆栈跟踪显示了Logger正在调用的实际播放方法。这可能是因为您试图在object这里模拟 Scala。相反,如果您像这样嘲笑特征

 val log = mock[play.api.LoggerLike]

您将克服此特定错误并进入下一个问题 - 您的验证不起作用:

[error]   The mock was not called as expected: 
[error] Argument(s) are different! Wanted:
[error] loggerLike.error(
[error]     ($anonfun$apply$mcV$sp$1) <function0>
[error] );
[error] -> at Service$$anonfun$verify$1.apply$mcV$sp(ServiceSpec.scala:10)
[error] Actual invocation has different arguments:
[error] loggerLike.error(
[error]     ($anonfun$fail$1) <function0>
[error] );
[error] -> at services.BaseService$class.fail(x.scala:57)
[error]  (ServiceSpec.scala:17)

这是由于日志消息作为按名称参数(因此是匿名函数)而不是字符串传递的事实引起的,请参阅此问题以进行进一步讨论。

于 2012-08-31T07:10:20.313 回答