Как протестировать клиентскую версию Akka HTTP

Я только что начал тестировать API-интерфейс клиентского уровня Akka HTTP (Future-Based). Одна вещь, которую я пытался понять, - это написать unit test для этого. Есть ли способ издеваться над ответом и завершить будущее без фактического выполнения HTTP-запроса?

Я смотрел API и пакет testkit, пытаясь понять, как я могу это использовать, только чтобы найти в документах, что он на самом деле говорит:

akka-http-testkit Испытательный жгут и набор утилит для проверки реализаций служб на стороне сервера

Я думал что-то TestServer (вроде как TestSource для потоков Akka), и используйте DSL на стороне сервера для создания ожидаемого ответа и каким-то образом подключите его к объекту Http.

Вот упрощенный пример того, что функция выполняет, которую я хочу проверить:

object S3Bucket {

  def sampleTextFile(uri: Uri)(
    implicit akkaSystem: ActorSystem,
    akkaMaterializer: ActorMaterializer
  ): Future[String] = {
    val request = Http().singleRequest(HttpRequest(uri = uri))
    request.map { response => Unmarshal(response.entity).to[String] }
  }
}

Ответ 1

Я думаю, что в общих чертах вы уже столкнулись с тем, что лучший подход состоит в том, чтобы высмеять ответ. В Scala это можно сделать, используя Scala Mock http://scalamock.org/

Если вы упорядочиваете свой код так, чтобы ваш экземпляр akka.http.scaladsl.HttpExt был введен в код, который его использует (например, как параметр конструктора), то во время тестирования вы можете ввести экземпляр mock[HttpExt], а не один из построенных с помощью метод применения Http.

EDIT: Я думаю, это было отклонено за то, что он недостаточно определен. Вот как я бы структурировал насмешку над вашим сценарием. Это сделалось немного сложнее всем имплицитом.

Код в main:

import akka.actor.ActorSystem
import akka.http.scaladsl.Http
import akka.http.scaladsl.model.{Uri, HttpResponse, HttpRequest}
import akka.http.scaladsl.unmarshalling.Unmarshal
import akka.stream.ActorMaterializer

import scala.concurrent.{ExecutionContext, Future}

trait S3BucketTrait {

  type HttpResponder = HttpRequest => Future[HttpResponse]

  def responder: HttpResponder

  implicit def actorSystem: ActorSystem

  implicit def actorMaterializer: ActorMaterializer

  implicit def ec: ExecutionContext

  def sampleTextFile(uri: Uri): Future[String] = {

    val responseF = responder(HttpRequest(uri = uri))
    responseF.flatMap { response => Unmarshal(response.entity).to[String] }
  }
}

class S3Bucket(implicit val actorSystem: ActorSystem, val actorMaterializer: ActorMaterializer) extends S3BucketTrait {

  override val ec: ExecutionContext = actorSystem.dispatcher

  override def responder = Http().singleRequest(_)
}

Код в test:

import akka.actor.ActorSystem
import akka.http.scaladsl.model._
import akka.stream.ActorMaterializer
import akka.testkit.TestKit
import org.scalatest.{BeforeAndAfterAll, WordSpecLike, Matchers}
import org.scalamock.scalatest.MockFactory
import scala.concurrent._
import scala.concurrent.duration._
import scala.concurrent.Future

class S3BucketSpec extends TestKit(ActorSystem("S3BucketSpec"))
with WordSpecLike with Matchers with MockFactory with BeforeAndAfterAll  {


  class MockS3Bucket(reqRespPairs: Seq[(Uri, String)]) extends S3BucketTrait{

    override implicit val actorSystem = system

    override implicit val ec = actorSystem.dispatcher

    override implicit val actorMaterializer = ActorMaterializer()(system)

    val mock = mockFunction[HttpRequest, Future[HttpResponse]]

    override val responder: HttpResponder = mock

    reqRespPairs.foreach{
      case (uri, respString) =>
        val req = HttpRequest(HttpMethods.GET, uri)
        val resp = HttpResponse(status = StatusCodes.OK, entity = respString)
        mock.expects(req).returning(Future.successful(resp))
    }
  }

  "S3Bucket" should {

    "Marshall responses to Strings" in {
      val mock = new MockS3Bucket(Seq((Uri("http://example.com/1"), "Response 1"), (Uri("http://example.com/2"), "Response 2")))
      Await.result(mock.sampleTextFile("http://example.com/1"), 1 second) should be ("Response 1")
      Await.result(mock.sampleTextFile("http://example.com/2"), 1 second) should be ("Response 2")
    }
  }

  override def afterAll(): Unit = {
    val termination = system.terminate()
    Await.ready(termination, Duration.Inf)
  }
}

build.sbt зависимости:

libraryDependencies += "com.typesafe.akka" % "akka-http-experimental_2.11" % "2.0.1"

libraryDependencies += "org.scalamock" %% "scalamock-scalatest-support" % "3.2" % "test"

libraryDependencies += "org.scalatest" % "scalatest_2.11" % "2.2.6"

libraryDependencies += "com.typesafe.akka" % "akka-testkit_2.11" % "2.4.1"

Ответ 2

Учитывая, что вы действительно хотите написать unit test для своего HTTP-клиента, вы должны притворяться, что нет реального сервера, а не пересекаете границу сети, иначе вы, очевидно, будете проводить интеграционные тесты. Длинный известный рецепт принудительного разделения, подлежащего тестированию, в таких случаях, как ваш, заключается в разделении интерфейса и реализации. Просто определите интерфейс, абстрагирующий доступ к внешнему HTTP-серверу и его реальным и поддельным реализациям, как в следующем эскизе

import akka.actor.Actor
import akka.pattern.pipe
import akka.http.scaladsl.HttpExt
import akka.http.scaladsl.model.{HttpRequest, HttpResponse, StatusCodes}
import scala.concurrent.Future

trait HTTPServer {
  def sendRequest: Future[HttpResponse]
}

class FakeServer extends HTTPServer {
  override def sendRequest: Future[HttpResponse] =
    Future.successful(HttpResponse(StatusCodes.OK))
}

class RealServer extends HTTPServer {

  def http: HttpExt = ??? //can be passed as a constructor parameter for example

  override def sendRequest: Future[HttpResponse] =
    http.singleRequest(HttpRequest(???))
}

class HTTPClientActor(httpServer: HTTPServer) extends Actor {

  override def preStart(): Unit = {
    import context.dispatcher
    httpServer.sendRequest pipeTo self
  }

  override def receive: Receive = ???
}

и протестируйте HTTPClientActor в сочетании с FakeServer.

Ответ 3

Я надеялся, что может быть способ использовать какую-то систему тестовых актеров, но в отсутствие этого (или какого-то другого идиоматического способа) я, вероятно, буду делать что-то вроде этого:

object S3Bucket {

  type HttpResponder = HttpRequest => Future[HttpResponse]

  def defaultResponder = Http().singleRequest(_)

  def sampleTextFile(uri: Uri)(
    implicit akkaSystem: ActorSystem,
    akkaMaterializer: ActorMaterializer,
    responder: HttpResponder = defaultResponder
  ): Future[String] = {
    val request = responder(HttpRequest(uri = uri))
    request.map { response => Unmarshal(response.entity).to[String] }
  }
}

Тогда в моем тесте я могу просто предоставить mock HttpResponder.