Route TestKit
One of Akka HTTP’s design goals is good testability of the created services. For services built with the Routing DSL Akka HTTP provides a dedicated testkit that makes efficient testing of route logic easy and convenient. This “route test DSL” is made available with the akka-http-testkit module.
Dependency
The Akka dependencies are available from Akka’s library repository. To access them there, you need to configure the URL for this repository.
To use Akka HTTP TestKit, add the module to your project:
- sbt
val AkkaVersion = "2.10.0" val AkkaHttpVersion = "10.7.0" libraryDependencies ++= Seq( "com.typesafe.akka" %% "akka-stream-testkit" % AkkaVersion, "com.typesafe.akka" %% "akka-http-testkit" % AkkaHttpVersion )
- Gradle
- Maven
Usage
To use the testkit you need to take these steps:
- add a dependency to the
akka-http-testkit
module - derive the test class from
JUnitRouteTest
- wrap the route under test with
RouteTest.testRoute
to create aTestRoute
- run requests against the route using
TestRoute.run(request)
which will return aTestResponse
- use the methods of
TestResponse
to assert on properties of the response
Example
To see the testkit in action consider the following simple calculator app service:
- Java
-
source
import akka.actor.ActorSystem; import akka.http.javadsl.Http; import akka.http.javadsl.server.AllDirectives; import akka.http.javadsl.server.Route; import akka.http.javadsl.server.examples.simple.SimpleServerApp; import akka.http.javadsl.unmarshalling.StringUnmarshallers; import java.io.IOException; public class MyAppService extends AllDirectives { public String add(double x, double y) { return "x + y = " + (x + y); } public Route createRoute() { return get(() -> pathPrefix("calculator", () -> path("add", () -> parameter(StringUnmarshallers.DOUBLE, "x", x -> parameter(StringUnmarshallers.DOUBLE, "y", y -> complete(add(x, y)) ) ) ) ) ); } public static void main(String[] args) throws IOException { final ActorSystem system = ActorSystem.create(); final SimpleServerApp app = new SimpleServerApp(); Http.get(system).newServerAt("127.0.0.1", 8080).bind(app.createRoute()); System.console().readLine("Type RETURN to exit..."); system.terminate(); } }
MyAppService
extends from AllDirectives
which brings all of the directives into scope. We define a method called createRoute
that provides the routes to serve to bind
.
Here’s how you would test that service:
- Java
-
source
import akka.http.javadsl.model.HttpRequest; import akka.http.javadsl.model.StatusCodes; import akka.http.javadsl.testkit.JUnitRouteTest; import akka.http.javadsl.testkit.TestRoute; import org.junit.Test; public class TestkitExampleTest extends JUnitRouteTest { TestRoute appRoute = testRoute(new MyAppService().createRoute()); @Test public void testCalculatorAdd() { // test happy path appRoute.run(HttpRequest.GET("/calculator/add?x=4.2&y=2.3")) .assertStatusCode(200) .assertEntity("x + y = 6.5"); // test responses to potential errors appRoute.run(HttpRequest.GET("/calculator/add?x=3.2")) .assertStatusCode(StatusCodes.NOT_FOUND) // 404 .assertEntity("Request is missing required query parameter 'y'"); // test responses to potential errors appRoute.run(HttpRequest.GET("/calculator/add?x=3.2&y=three")) .assertStatusCode(StatusCodes.BAD_REQUEST) .assertEntity("The query parameter 'y' was malformed:\n" + "'three' is not a valid 64-bit floating point value"); } }
Writing Asserting against the HttpResponse
The testkit supports a fluent DSL to write compact assertions on the response by chaining assertions using “dot-syntax”. To simplify working with streamed responses the entity of the response is first “strictified”, i.e. entity data is collected into a single ByteString
and provided the entity is supplied as an HttpEntityStrict
. This allows to write several assertions against the same entity data which wouldn’t (necessarily) be possible for the streamed version.
All of the defined assertions provide HTTP specific error messages aiding in diagnosing problems.
Currently, these methods are defined on TestResponse
to assert on the response:
Inspector | Description |
---|---|
assertStatusCode(int expectedCode) |
Asserts that the numeric response status code equals the expected one |
assertStatusCode(StatusCode expectedCode) |
Asserts that the response StatusCode equals the expected one |
assertMediaType(String expectedType) |
Asserts that the media type part of the response’s content type matches the given String |
assertMediaType(MediaType expectedType) |
Asserts that the media type part of the response’s content type matches the given MediaType |
assertEntity(String expectedStringContent) |
Asserts that the entity data interpreted as UTF8 equals the expected String |
assertEntityBytes(ByteString expectedBytes) |
Asserts that the entity data bytes equal the expected ones |
assertEntityAs(Unmarshaller<T> unmarshaller, expectedValue: T) |
Asserts that the entity data if unmarshalled with the given marshaller equals the given value |
assertHeaderExists(HttpHeader expectedHeader) |
Asserts that the response contains an HttpHeader instance equal to the expected one |
assertHeaderKindExists(String expectedHeaderName) |
Asserts that the response contains a header with the expected name |
assertHeader(String name, String expectedValue) |
Asserts that the response contains a header with the given name and value. |
It’s, of course, possible to use any other means of writing assertions by inspecting the properties the response manually. As written above, TestResponse.entity
and TestResponse.response
return strict versions of the entity data.
Supporting Custom Test Frameworks
Adding support for a custom test framework is achieved by creating new superclass analogous to JUnitRouteTest
for writing tests with the custom test framework deriving from akka.http.javadsl.testkit.RouteTest
and implementing its abstract methods. This will allow users of the test framework to use testRoute
and to write assertions using the assertion methods defined on TestResponse
.
Testing sealed Routes
The section above describes how to test a “regular” branch of your route structure, which reacts to incoming requests with HTTP response parts or rejections. Sometimes, however, you will want to verify that your service also translates Rejections to HTTP responses in the way you expect.
You do this by calling the Route.seal()
method. The Route.seal()
method applies the logic of the ExceptionHandler and RejectionHandler passed as method arguments to all exceptions and rejections coming back from the route, and translates them to the respective HttpResponse
.
Note that explicit call on the Route.seal
method is needed in test code, but in your application code it is not necessary. As described in Sealing a Route, your application code only needs to bring implicit rejection and exception handlers in scope.
Testing Route fragments
Since the testkit is request-based, you cannot test requests that are illegal or impossible in HTTP. One such instance is testing a route that begins with the pathEnd
directive, such as routeFragment
here:
You might create a route such as this to be able to compose it into another route such as:
- Scala
- Java
-
source
import akka.http.javadsl.server.AllDirectives; import akka.http.javadsl.server.Route; public class MyAppFragment extends AllDirectives { public Route createRoute() { return pathEnd(() -> get(() -> complete("Fragments of imagination") ) ); } }
However, it is impossible to unit test this Route directly using testkit, since it is impossible to create an empty HTTP request. To test this type of route, embed it in a synthetic route in your test, such as testRoute
in the example above.
This is what the full working test looks like:
- Scala
- Java
-
source
import akka.http.javadsl.model.HttpRequest; import akka.http.javadsl.model.StatusCodes; import akka.http.javadsl.server.AllDirectives; import akka.http.javadsl.server.Route; import akka.http.javadsl.testkit.JUnitRouteTest; import akka.http.javadsl.testkit.TestRoute; import org.junit.Test; public class TestKitFragmentTest extends JUnitRouteTest { class FragmentTester extends AllDirectives { public Route createRoute(Route fragment) { return pathPrefix("test", () -> fragment ); } } TestRoute fragment = testRoute(new MyAppFragment().createRoute()); TestRoute testRoute = testRoute(new FragmentTester().createRoute(fragment.underlying())); @Test public void testFragment() { testRoute.run(HttpRequest.GET("/test")) .assertStatusCode(200) .assertEntity("Fragments of imagination"); testRoute.run(HttpRequest.PUT("/test")) .assertStatusCode(StatusCodes.METHOD_NOT_ALLOWED); } }
Accounting for Slow Test Systems
The timeouts you consciously defined on your lightning fast development environment might be too tight for your, most probably, high-loaded Continuous Integration server, invariably causing spurious failures. To account for such situations, timeout durations can be scaled by a given factor on such environments. Check the Akka Docs for further information.
Increase Timeout
The default timeout when testing your routes using the testkit is 3 seconds second. Sometimes, though, this might not be enough. In order to extend this default timeout, to say 5 seconds, just add the following implicit in scope:
- Scala
- Java
-
source
@Override public FiniteDuration awaitDuration() { return FiniteDuration.create(5, TimeUnit.SECONDS); }
Remember to configure the timeout using dilated
if you want to account for slow test systems.
Testing Actor integration
The JUnitRouteTest
still provides a Classic ActorSystem
, so if you are not using the Classic API you will need to adapt it:
- Scala
- Java
-
source
import akka.actor.testkit.typed.javadsl.TestProbe; import akka.actor.typed.ActorSystem; import akka.actor.typed.javadsl.Adapter; import akka.http.javadsl.model.HttpRequest; import akka.http.javadsl.testkit.JUnitRouteTest; import akka.http.javadsl.testkit.TestRoute; import akka.http.javadsl.testkit.TestRouteResult; import org.junit.Test; public class TestKitWithActorTest extends JUnitRouteTest { @Test public void returnPongForGetPing() { // This test does not use the classic APIs, // so it needs to adapt the system: ActorSystem<Void> system = Adapter.toTyped(system()); TestProbe<MyAppWithActor.Ping> probe = TestProbe.create(system); TestRoute testRoute = testRoute(new MyAppWithActor().createRoute(probe.getRef(), system.scheduler())); TestRouteResult result = testRoute.run(HttpRequest.GET("/ping")); MyAppWithActor.Ping ping = probe.expectMessageClass(MyAppWithActor.Ping.class); ping.replyTo.tell("PONG!"); result.assertEntity("PONG!"); } }
Integration Testing Routes
Use ~!>
to test a route running in full HTTP server mode:
REQUEST ~!> ROUTE ~> check {
ASSERTIONS
}
Certain routes can only be tested with ~!>
, for example routes that use the withRequestTimeout
directive.
Using ~!>
adds considerable extra overhead since each test will start a server and bind to a port so use it only when necessary.
Examples
A great pool of examples are the tests for all the predefined directives in Akka HTTP. They can be found here.