Quarkus - Quarkus Extension for Spring Web API

While users are encouraged to use JAX-RS annotation for defining REST endpoints, Quarkus provides a compatibility layer for Spring Web in the form of the spring-web extension.

This guide explains how a Quarkus application can leverage the well known Spring Web annotations to define RESTful services.

Prerequisites

To complete this guide, you need:

  • less than 15 minutes

  • an IDE

  • JDK 1.8+ installed with JAVA_HOME configured appropriately

  • Apache Maven 3.5.3+

Solution

We recommend that you follow the instructions in the next sections and create the application step by step.However, you can go right to the completed example.

Clone the Git repository: git clone https://github.com/quarkusio/quarkus-quickstarts.git, or download an archive.

The solution is located in the spring-web-quickstart directory.

Creating the Maven project

First, we need a new project. Create a new project with the following command:

  1. mvn io.quarkus:quarkus-maven-plugin:1.0.0.CR1:create \
  2. -DprojectGroupId=org.acme \
  3. -DprojectArtifactId=spring-web-quickstart \
  4. -DclassName="org.acme.spring.web.GreetingController" \
  5. -Dpath="/greeting" \
  6. -Dextensions="spring-web"
  7. cd spring-web-quickstart

This command generates a Maven project with a REST endpoint and imports the spring-web extension.

GreetingController

The Quarkus maven plugin automatically generated a controller with the Spring Web annotations to define our REST endpoint (instead of the JAX-RS ones used by default)The src/main/java/org/acme/spring/web/GreetingController.java file looks as follows:

  1. package org.acme.spring.web;
  2. import org.springframework.web.bind.annotation.GetMapping;
  3. import org.springframework.web.bind.annotation.RequestMapping;
  4. import org.springframework.web.bind.annotation.RestController;
  5. import org.springframework.web.bind.annotation.PathVariable;
  6. @RestController
  7. @RequestMapping("/greeting")
  8. public class GreetingController {
  9. @GetMapping
  10. public String hello() {
  11. return "hello";
  12. }
  13. }

GreetingControllerTest

Note that a test for the controller has been created as well:

  1. package org.acme.spring.web;
  2. import io.quarkus.test.junit.QuarkusTest;
  3. import org.junit.jupiter.api.Test;
  4. import static io.restassured.RestAssured.given;
  5. import static org.hamcrest.CoreMatchers.is;
  6. @QuarkusTest
  7. public class GreetingControllerTest {
  8. @Test
  9. public void testHelloEndpoint() {
  10. given()
  11. .when().get("/greeting")
  12. .then()
  13. .statusCode(200)
  14. .body(is("hello"));
  15. }
  16. }

Package and run the application

Run the application with: ./mvnw compile quarkus:dev.Open your browser to http://localhost:8080/greeting.

The result should be: {"message": "hello"}.

Run the application as a native executable

You can of course create a native image using the instructions of the Building a native executable guide.

Going further with an endpoint returning JSON

The GreetingController above was an example of a very simple endpoint. In many cases however it is required to return JSON content.The following example illustrates how that could be achieved using a Spring RestController:

  1. import org.springframework.web.bind.annotation.GetMapping;
  2. import org.springframework.web.bind.annotation.PathVariable;
  3. import org.springframework.web.bind.annotation.RequestMapping;
  4. import org.springframework.web.bind.annotation.RestController;
  5. @RestController
  6. @RequestMapping("/greeting")
  7. public class GreetingController {
  8. @GetMapping("/{name}")
  9. public Greeting hello(@PathVariable(name = "name") String name) {
  10. return new Greeting("hello " + name);
  11. }
  12. public static class Greeting {
  13. private final String message;
  14. public Greeting(String message) {
  15. this.message = message;
  16. }
  17. public String getMessage(){
  18. return message;
  19. }
  20. }
  21. }

The corresponding test could look like:

  1. package org.acme.spring.web;
  2. import io.quarkus.test.junit.QuarkusTest;
  3. import org.junit.jupiter.api.Test;
  4. import static io.restassured.RestAssured.given;
  5. import static org.hamcrest.CoreMatchers.is;
  6. @QuarkusTest
  7. public class GreetingControllerTest {
  8. @Test
  9. public void testHelloEndpoint() {
  10. given()
  11. .when().get("/greeting/quarkus")
  12. .then()
  13. .statusCode(200)
  14. .body("message", is("hello quarkus"));
  15. }
  16. }

It should be noted that when using the Spring Web support in Quarkus, Jackson is automatically added to the classpath and properly setup.

Supported Spring Web functionalities

Quarkus currently supports a subset of the functionalities that Spring Web provides. More specifically Quarkus supports the REST related features of Spring Web(think of @RestController instead of @Controller).

Annotations

The table below summarizes the supported annotations:

Table 1. Supported Spring Web annotation
NameComments
@RestController
@RequestMapping
@GetMapping
@PostMapping
@PutMapping
@DeleteMapping
@PatchMapping
@RequestParam
@RequestHeader
@MatrixVariable
@PathVariable
@CookieValue
@RequestBody
@ResponseStatus
@ExceptionHandlerCan only be used in a @RestControllerAdvice class, not on a per-controller basis
@RestControllerAdviceOnly the @ExceptionHandler capability is supported

Controller method return types

The following method return types are supported:

  • Primitive types

  • String (which will be used as a literal, no Spring MVC view support is provided)

  • POJO classes which will be serialized via JSON

  • org.springframework.http.ResponseEntity

Controller method return types

In addition to the method parameters that can be annotated with the appropriate Spring Web annotations from the previous table,javax.servlet.http.HttpServletRequest and javax.servlet.http.HttpServletResponse are also supported.For this to function however, users need to add the quarkus-undertow dependency.

Important Technical Note

Please note that the Spring support in Quarkus does not start a Spring Application Context nor are any Spring infrastructure classes run.Spring classes and annotations are only used for reading metadata and / or are used as user code method return types or parameter types.What that means for end users, is that adding arbitrary Spring libraries will not have any effect. Moreover Spring infrastructureclasses (like org.springframework.beans.factory.config.BeanPostProcessor for example) will not be executed.

More Spring guides

Quarkus support has more Spring compatibility features. See the following guides for more details: