2

我正在尝试编写一些功能测试,并且我想模拟一个使用外部提供程序的服务。但我无法为返回的函数设置模拟EitherT

这是其实现调用外部服务的特征

@ImplementedBy(classOf[ExternalServiceImpl])
trait ExternalService {
  def index: EitherT[Future, String, JsValue]
}

在我设置的 CustomAppPerSuite 特征中

val mockExternalService = mock[ExternalService]

 implicit override lazy val app = new GuiceApplicationBuilder()
.in(Mode.Test)
.overrides(bind[ExternalService].toInstance(mockExternalService))
.build()

val externalService = app.injector.instanceOf[ExternalService]

然后当我尝试模拟成功的响应时

  "ExternalController#index" should {

    "return index data" in {
      doReturn(EitherT.rightT(Json.parse("""{"key": "value"}""")).when(externalService).index
      val fakeRequest = FakeRequest(GET, "/api/external")
      val result = externalController.index().apply(fakeRequest)
      status(result) mustBe OK
    }

但我得到这个错误

[error]  found   : cats.data.EitherT[cats.package.Id,Nothing,JsValue]
[error]  required: cats.data.EitherT[scala.concurrent.Future,String,JsValue]
[error]   def index = EitherT.rightT(

我只想嘲笑成功的响应,因为这是我正在测试的。有没有办法做到这一点?

4

2 回答 2

3

使用 mockito-scala-cats 你可以用更简洁的方式来写

Json.parse("""{"key": "value"}""") willBe returnedF by externalService.index
//or
externalService.index shouldReturnF Json.parse("""{"key": "value"}""")

库将查看的返回类型externalService.index并获取适当cats.Applicative的 (s) 以使这项工作顺利进行。

如果您在 Scalatest 上运行,另一个优势是您可以混合ResetMocksAfterEachTest并让您连接到 play fake 应用程序的所有模拟在每次测试之前自动重置。

在这里查看更多详情

于 2019-05-21T09:52:08.140 回答
2

尝试通过提供一些rightT喜欢的类型参数来帮助编译器

EitherT.rightT[Future, String](Json.parse("""{"key": "value"}"""))

代替

EitherT.rightT(Json.parse("""{"key": "value"}"""))
于 2019-05-16T15:20:04.030 回答