Javalin is very flexible and allows you to extend it in many ways. This repository contains a set of extensions for Javalin routing system following some of the most popular patterns. Each approach has pros and cons, so you should choose the one that fits your needs the best.
Each module is distributed as a separate artifact:
dependencies {
val javalinRoutingExtensions = "6.4.0"
implementation("io.javalin.community.routing:routing-core:$javalinRoutingExtensions")
implementation("io.javalin.community.routing:routing-annotated:$javalinRoutingExtensions")
implementation("io.javalin.community.routing:routing-dsl:$javalinRoutingExtensions")
implementation("io.javalin.community.routing:routing-coroutines:$javalinRoutingExtensions")
}
This chapter provides short preview of each module. For more details, please refer to the documentation or full example of each module. First of all, not each module is available for Java users, take a look on the table below to check requirements:
Module | Languages | Reflections |
---|---|---|
Annotated | Java, Kotlin | Yes (as long as we won't provide annotation processor) |
DSL In-place DSL Properties |
Kotlin | Optional |
Coroutines | Kotlin | No |
Core | Java, Kotlin | No |
This module provides set of annotations to simplify routing setup & basic http operations. This is probably the most common approach to routing in Java world, some people may even say that it's the only one. Take a look on the example below to see how it looks like:
import static io.javalin.community.routing.annotations.AnnotatedRouting.Annotated;
// register endpoints with prefix
@Endpoints("/api")
static final class ExampleEndpoints {
private final ExampleService exampleService;
// pass dependencies required to handle requests
public ExampleEndpoints(ExampleService exampleService) {
this.exampleService = exampleService;
}
// use Javalin-specific routes
@Before
void beforeEach(Context ctx) {
ctx.header("X-Example", "Example");
}
// describe http method and path with annotation
@Post("/hello")
// use parameters to extract data from request
void saveExample(Context context, @Nullable @Header(AUTHORIZATION) String authorization, @Body ExampleDto entity) {
if (authorization == null) {
context.status(401);
return;
}
exampleService.saveExample(entity);
}
// you can combine it with OpenApi plugin
@OpenApi(
path = "/api/hello/{name}",
methods = { GET },
summary = "Find example by name",
pathParams = { @OpenApiParam(name = "name", description = "Name of example to find") },
responses = { @OpenApiResponse(status = "200", description = "Example found", content = @OpenApiContent(from = ExampleDto.class)) },
versions = { "default", "2" }
)
// you can also use out-of-the-box support for versioned routes
@Version("2")
@Get("/hello/{name}")
void findExampleV2(Context context, @Param String name) {
context.result(exampleService.findExampleByName(name).name);
}
/* OpenApi [...] */
@Version("1")
@Get("/hello/{name}")
void findExampleV1(Context ctx) {
throw new UnsupportedOperationException("Deprecated");
}
// register exception handlers alongside endpoints
@ExceptionHandler(Exception.class)
void defaultExceptionHandler(Exception e, Context ctx) {
ctx.status(500).result("Something went wrong: " + e.getClass());
}
}
public static void main(String[] args) {
Javalin.createAndStart(config -> {
// prepare dependencies
var exampleService = new ExampleService();
// register endpoints
config.router.mount(Annotated, routing -> {
routing.registerEndpoints(new ExampleEndpoints(exampleService));
});
});
// test request to `saveExample` endpoint
HttpResponse<?> saved = Unirest.post("http://localhost:7000/api/hello")
.basicAuth("Panda", "passwd")
.body(new ExampleDto("Panda"))
.asEmpty();
System.out.println("Entity saved: " + saved.getStatusText()); // Entity saved: OK
// test request to `findExampleV2` endpoint
String result = Unirest.get("http://localhost:7000/api/hello/Panda")
.header("X-API-Version", "2")
.asString()
.getBody();
System.out.println("Entity: " + result); // Entity: Panda
}
This approach requires some reflections under the hood to make work at this moment, but we're working on annotation processor to remove this requirement.
Another thing you can notice is that we're creating endpoint class instance using constructor, not built-in DI framework. This is because in general we consider this as a good practice - not only because you're in full control over the execution flow, but also because it forces you to make concious decision about the scope of your dependencies & architecture.
If you don't really care about it, and you're just looking for a tool that will get the job done, you can use literally any DI framework you want that is available for Java/Kotlin. We may recommend Dagger2, because it verifies your code at compile time, so it's safer than heavy reflection-based alternatives.
Keep in mind, that if you want to use named parameters in your endpoints,
you have to pass -parameters
flag to your compiler to preserve parameter names in bytecode.
Full example: AnnotationsRoutingExample.java
DSL provides extensible base for creating custom DSLs for routing.
By default, this module provides basic implementation of Ktor-like routing for Kotlin apps with support for type-safe paths:
@Path("/panda/{age}")
data class PandaPath(val age: Int)
fun main() {
Javalin.create { config ->
config.routing(CustomDsl) {
before {
// `endpointHandlerPath` comes from Context class
result("Called endpoint: ${matchedPath()}")
}
get("/") {
// `helloWorld` comes from CustomScope class
result(helloWorld())
}
get<PandaPath> { path ->
// support for type-safe paths
result(path.age.toString())
}
exception(Exception::class) { anyException ->
// support for exception handlers
result(anyException.message ?: "Unknown error")
}
}
}.start(8080)
}
Because of the extensible nature of DSL, you may adjust it to your needs! You can find base implementation of custom DSL definition here: InPlaceExample.kt
Property based implementation of DSL allows you to easily define routes in multiple sources,
outside the main setup scope.
This approach is very similar to Spring Boot's @RestController
annotation,
but it also supports your custom type-safe DSL, and
you're in full control of your execution flow.
// Some dependencies
class ExampleService {
fun save(animal: String) = println("Saved animal: $animal")
}
// Utility representation of custom routing in your application
abstract class ExampleRouting : DslRoutes<DslRoute<CustomScope, Unit>, CustomScope, Unit>
// Endpoint (domain router)
class AnimalEndpoints(private val exampleService: ExampleService) : ExampleRouting() {
@OpenApi(
path = "/animal/{name}",
methods = [HttpMethod.GET]
)
private val findAnimalByName = route("/animal/<name>", GET) {
result(pathParam("name"))
}
@OpenApi(
path = "/animal/{name}",
methods = [HttpMethod.POST]
)
private val saveAnimal = route("/animal/<name>", POST) {
exampleService.save(pathParam("name"))
}
private val defaultExceptionHandler = exceptionHandler(Exception::class) { regularException ->
println("Exception: ${regularException.message}")
}
override fun routes() = setOf(findAnimalByName, saveAnimal)
override fun exceptionHandlers() = setOf(defaultExceptionHandler)
}
fun main() {
// prepare dependencies
val exampleService = ExampleService()
// setup & launch application
Javalin
.create { it.routing(CustomDsl, AnimalEndpoints(exampleService) /*, provide more classes with endpoints */) }
.start(8080)
}
This example is based on previous in-place example, you can check its source code here: PropertyDslExample.kt
Experimental: This module is more like a reference on how to use coroutines with Javalin. The production-readiness of this module is unknown, especially in complex scenarios.
The coroutines module provides API similar to DSL :: Properties
module,
but it uses coroutines & suspend directives to provide asynchronous & non-blocking endpoint execution.
// Custom scope used by routing DSL
class CustomScope(val ctx: Context) : Context by ctx {
// blocks thread using reactive `delay` function
suspend fun nonBlockingDelay(message: String): String = delay(2000L).let { message }
}
// Utility class representing group of reactive routes
abstract class ExampleRoutes : ReactiveRoutes<ReactiveRoute<CustomScope, Unit>, CustomScope, Unit>()
// Endpoint (domain router)
class ExampleEndpoint(private val exampleService: ExampleService) : ExampleRoutes() {
// you can use suspend functions in coroutines context
// and as long as they're truly reactive, they won't freeze it
private val nonBlockingAsync = reactiveRoute("/async", GET) {
result(nonBlockingDelay("Non-blocking Async"))
}
override fun routes() = setOf(nonBlockingAsync)
}
fun main() {
// prepare dependencies
val exampleService = ExampleService()
// create coroutines servlet with single-threaded executor
val coroutinesServlet = DefaultContextCoroutinesServlet(
executorService = Executors.newSingleThreadExecutor(),
contextFactory = { CustomScope(it) },
)
// setup Javalin with reactive routing
Javalin
.create { config ->
config.reactiveRouting(coroutinesServlet, ExampleEndpoint(exampleService))
}
.events {
it.serverStopping { coroutinesServlet.prepareShutdown() }
it.serverStopped { coroutinesServlet.completeShutdown() }
}
.start("127.0.0.1", 8080)
}
Full example: ReactiveRoutingExample.kt
The core module contains shared components for the other modules.
The most important functionality is the RouteComparator
which is used to sort given set of routes in the correct order by associated route path.
- Reposilite - real world app using Javalin with property-based DSL routing