Skip to content

Latest commit

 

History

History
341 lines (273 loc) · 13.1 KB

README.md

File metadata and controls

341 lines (273 loc) · 13.1 KB

Javalin Routing Extensions Plugin

CI Coverage Maven Central Version / Snapshot

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.

  1. Installation
  2. Usage
    1. Annotated
    2. DSL
      1. In-place
      2. Properties
    3. Coroutines
    4. Core

Installation

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")
}

Usage

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

Annotated

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

DSL provides extensible base for creating custom DSLs for routing.

In-place

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

Properties

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

Coroutines

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

Core

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.

Other examples

  • Reposilite - real world app using Javalin with property-based DSL routing