| CI | Release | Issues | Snapshot |
|---|---|---|---|
Akka-HTTP route TestKit for zio-test
libraryDependencies += "info.senia" %% "zio-test-akka-http" % "x.x.x"The basic structure of a test built with the testkit is this (expression placeholder in all-caps):
(REQUEST ~> ROUTE).map { res =>
assertTrue(res.some.path == value)
}
See RouteZIOSpecDefaultSpec.scala for usage examples.
Import zio.test.akkahttp.assertions._ and provide RouteTestEnvironment to use this testkit:
import akka.http.scaladsl.server.Directives.complete
import akka.http.scaladsl.model.HttpResponse
import zio.test.Assertion._
import zio.test._
import zio.test.akkahttp.RouteTestEnvironment
import zio.test.akkahttp.assertions._
object MySpec extends ZIOSpecDefault {
def spec =
suite("MySpec")(
test("my test") {
(Get() ~> complete(HttpResponse())).map { res =>
assertTrue(res.handled.get.response == HttpResponse())
}
}
).provideShared(RouteTestEnvironment.environment)
}Alternatively you can use DefaultAkkaRunnableSpec without importing assertions._ and providing additional environment:
import akka.http.scaladsl.server.Directives.complete
import akka.http.scaladsl.model.HttpResponse
import zio.test.Assertion._
import zio.test._
import zio.test.akkahttp.AkkaZIOSpecDefault
object MySpec extends AkkaZIOSpecDefault {
def spec =
suite("MySpec")(
test("my test") {
(Get() ~> complete(HttpResponse())).map { res =>
assertTrue(res.handled.get.response == HttpResponse())
}
}
)
}You can use HttpRequest or ZIO[R, E, HttpRequest] as request:
import akka.http.scaladsl.model.HttpResponse
import akka.http.scaladsl.server.Directives.complete
import zio.test.Assertion._
import zio.test._
import zio.test.akkahttp.assertions._
(Get() ~> complete(HttpResponse())).map { res =>
assertTrue(res.handled.get.response == HttpResponse())
}Available request builders: Get,Post,Put,Patch,Delete,Options,Head.
You can use any function HttpRequest => HttpRequest to modify request.
There are several common request modifications provided: addHeader,mapHeaders,removeHeader,addCredentials:
Get() ~> addHeader("MyHeader", "value") ~> routeYou can also add header with ~> method:
Get() ~> RawHeader("MyHeader", "value") ~> routeThere are several methods on response available:
handledresponseresponseEntitychunksentityAsresponseAscontentTypemediaTypecharsetheadersheaderstatusclosingExtensiontrailerrejectedisWebSocketUpgradewebSocketUpgradeProtocol
There should be a method for every inspector from original Akka-HTTP Route TestKit. If you can't find a method for existing inspector please open an issue.
Note that some methods are available only on eager version of response, but not on lazy one. You can always convert lazy response to eager with toEager method.
It's sage to call get on Option inside of asertTrue macros.
Note that response is eager on response fetching by default - you can't use ~> for status code and headers if route returns an infinite body.
To avoid eager response body fetching use lazy ?~> method instead of the last ~>:
import akka.http.scaladsl.model.StatusCodes.OK
import akka.http.scaladsl.server.Directives.{complete, get, respondWithHeader}
import akka.http.scaladsl.model.{ContentTypes, HttpEntity, HttpResponse}
import akka.http.scaladsl.model.headers.RawHeader
import akka.http.scaladsl.server.Directives
import akka.stream.scaladsl.Source
import akka.util.ByteString
import zio.test.Assertion._
import zio.test._
import zio.test.akkahttp.RouteTestEnvironment
import zio.test.akkahttp.assertions._
val pinkHeader = RawHeader("Fancy", "pink")
val route = get {
respondWithHeader(pinkHeader) {
complete(HttpEntity(ContentTypes.`application/octet-stream`, Source.repeat(ByteString("abc"))))
}
}
(Get() ?~> route).map { res =>
assertTrue(
res.handled.get.status == OK,
res.handled.get.header("Fancy").get == pinkHeader,
)
}