Slide 1

Slide 1 text

Matt Raible | @mraible June 22, 2023 Native Java REST API Comparison Micronaut, Quarkus, Spring Boot, and Helidon Photo by Leon Seibert https://unsplash.com/photos/xj_x5FD9V08

Slide 2

Slide 2 text

@mraible Who is Matt Raible? Father, Husband, Skier, Mountain Biker, Whitewater Rafter Bus Lover Web Developer and Java Champion Okta Developer Advocate Blogger on raibledesigns.com and developer.okta.com/blog @mraible

Slide 3

Slide 3 text

No content

Slide 4

Slide 4 text

No content

Slide 5

Slide 5 text

No content

Slide 6

Slide 6 text

developer.okta.com

Slide 7

Slide 7 text

developer.auth0.com

Slide 8

Slide 8 text

@mraible Today’s Agenda Why Java? Build { REST, GraphQL } APIs with Java Secure your APIs with OAuth 2.1 Build with Docker Go Native with GraalVM https://unsplash.com/photos/JsTmUnHdVYQ

Slide 9

Slide 9 text

@mraible Why Java? 25+ Years of use, abuse, and improvements Open Source code is available; many popular open source frameworks and tools Hugely Popular and widely used by many enterprises and web-scale companies

Slide 10

Slide 10 text

@mraible Download the Oracle builds of OpenJDK https://jdk.java.net/20 Or Eclipse builds from Adoptium https://adoptium.net Get Started with Java 20

Slide 11

Slide 11 text

@mraible Get Started with Java 20 Better yet, use SDKMAN! curl -s https://get.sdkman.io | bash 
 sdk install java 20-open

Slide 12

Slide 12 text

Java Releases and Features

Slide 13

Slide 13 text

https://developer.okta.com/blog/2020/01/09/java-rest-api-showdown Build REST APIs with Java

Slide 14

Slide 14 text

https://developer.okta.com/blog/2021/06/18/native-java-framework-comparison Build Native Java REST APIs

Slide 15

Slide 15 text

https://developer.okta.com/blog/2022/01/06/native-java-helidon Build REST APIs and Native Apps with Helidon

Slide 16

Slide 16 text

Serverless 💵 💸 https://unsplash.com/photos/glRqyWJgUeY

Slide 17

Slide 17 text

Use Cases for Native Images https://twitter.com/therealdanvega/status/1666098085616046081

Slide 18

Slide 18 text

@mraible sdk install micronaut mn create-app com.okta.rest.app \ -b maven -f security-jwt -f micronaut-aot Get Started with Micronaut

Slide 19

Slide 19 text

https://micronaut.io/launch

Slide 20

Slide 20 text

package com.okta.rest.controller; import io.micronaut.http.MediaType; import io.micronaut.http.annotation.Controller; import io.micronaut.http.annotation.Get; import io.micronaut.http.annotation.Produces; import io.micronaut.security.annotation.Secured; import io.micronaut.security.rules.SecurityRule; import java.security.Principal; @Controller("/hello") public class HelloController { @Get @Secured(SecurityRule.IS_AUTHENTICATED) @Produces(MediaType.TEXT_PLAIN) public String hello(Principal principal) { return "Hello, " + principal.getName() + "!"; } }

Slide 21

Slide 21 text

micronaut.security.token.jwt.enabled=true micronaut.security.token.jwt.signatures.jwks.auth0.url =https://dev-1337.us.auth0.com/.well-known/jwks.json Micronaut JWT Security

Slide 22

Slide 22 text

micronaut.security.token.jwt.enabled=true micronaut.security.token.jwt.signatures.jwks.auth0.url =https://dev-1337.us.auth0.com/.well-known/jwks.json Micronaut JWT Security https://micronaut-projects.github.io/micronaut-security/latest/guide/#jwt

Slide 23

Slide 23 text

Install HTTPie (a better cURL) $ install httpie https://httpie.org

Slide 24

Slide 24 text

Test Micronaut with HTTPie https://httpie.org mvn mn:run http :8080/hello TOKEN=eyJraWQiOiJxOE1QMjFNNHZCVmxOSkxGbFFWNlN... http :8080/hello Authorization:"Bearer $TOKEN"

Slide 25

Slide 25 text

Verify Micronaut API with HTTPie

Slide 26

Slide 26 text

@mraible Get Started with Quarkus sdk install quarkus quarkus create app com.okta.rest:quarkus \ --extension="smallrye-jwt,resteasy"

Slide 27

Slide 27 text

@mraible mvn io.quarkus:quarkus-maven-plugin:3.1.2.Final:create \ -DprojectGroupId=com.okta.rest \ -DprojectArtifactId=quarkus \ -DclassName="com.okta.rest.HelloResource" \ -Dpath="/hello" \ -Dextensions="smallrye-jwt,resteasy" Get Started with Quarkus

Slide 28

Slide 28 text

https://code.quarkus.io

Slide 29

Slide 29 text

package com.okta.rest; import io.quarkus.security.Authenticated; import jakarta.ws.rs.GET; import jakarta.ws.rs.Path; import jakarta.ws.rs.Produces; import jakarta.ws.rs.core.Context; import jakarta.ws.rs.core.MediaType; import jakarta.ws.rs.core.SecurityContext; import java.security.Principal; @Path("/hello") public class HelloResource { @GET @Authenticated @Produces(MediaType.TEXT_PLAIN) public String hello(@Context SecurityContext context) { Principal userPrincipal = context.getUserPrincipal(); return "Hello, " + userPrincipal.getName() + "!"; } }

Slide 30

Slide 30 text

mp.jwt.verify.issuer=https:// dev-1337.us.auth0.com mp.jwt.verify.publickey.location= ${mp.jwt.verify.issuer}/.well- known/jwks.json MicroProfile JWT Security https://www.eclipse.org/community/eclipse_newsletter/2017/september/article2.php

Slide 31

Slide 31 text

mp.jwt.verify.issuer=https:// dev-1337.us.auth0.com mp.jwt.verify.publickey.location= ${mp.jwt.verify.issuer}/.well- known/jwks.json MicroProfile JWT Security https://www.eclipse.org/community/eclipse_newsletter/2017/september/article2.php

Slide 32

Slide 32 text

No content

Slide 33

Slide 33 text

Test Quarkus with HTTPie https://httpie.org mvn quarkus:dev http :8080/hello TOKEN=eyJraWQiOiJxOE1QMjFNNHZCVmxOSkxGbFFWNlN... http :8080/hello Authorization:"Bearer $TOKEN"

Slide 34

Slide 34 text

Verify Quarkus API with HTTPie

Slide 35

Slide 35 text

@mraible Get Started with Spring Boot https start.spring.io/starter.zip \ dependencies==web,oauth2-resource-server,native \ packageName==com.okta.rest \ name==spring-boot \ type==maven-project \ baseDir==spring-boot | tar -xzvf -

Slide 36

Slide 36 text

https://start.spring.io https://start.spring.io

Slide 37

Slide 37 text

@mraible Use the Spring Boot CLI sdk install springboot spring init -d=web,oauth2-resource-server,native \ --package-name=com.okta.rest spring-boot \ --build=maven

Slide 38

Slide 38 text

package com.okta.rest.controller; import org.springframework.web.bind.annotation.GetMapping; import org.springframework.web.bind.annotation.RestController; import java.security.Principal; @RestController public class HelloController { @GetMapping("/hello") public String hello(Principal principal) { return "Hello, " + principal.getName() + "!"; } }

Slide 39

Slide 39 text

Spring Security OAuth 2.0 Resource Server https://docs.spring.io/spring-security/reference/servlet/oauth2/resource-server okta.oauth2.issuer=https://dev-1337.us.auth0.com

Slide 40

Slide 40 text

Test Spring Boot with HTTPie https://httpie.org mvn spring-boot:run http :8080/hello TOKEN=eyJraWQiOiJxOE1QMjFNNHZCVmxOSkxGbFFWNlN... http :8080/hello Authorization:"Bearer $TOKEN"

Slide 41

Slide 41 text

Verify Spring Boot API with HTTPie

Slide 42

Slide 42 text

@mraible Get Started with Helidon mvn -U archetype:generate -DinteractiveMode=false \ -DarchetypeGroupId=io.helidon.archetypes \ -DarchetypeArtifactId=helidon-quickstart-mp \ -DarchetypeVersion=3.2.1 \ -DgroupId=com.okta.rest \ -DartifactId=helidon \ -Dpackage=com.okta.rest

Slide 43

Slide 43 text

https://helidon.io/starter

Slide 44

Slide 44 text

Use the Helidon CLI helidon init --flavor MP \ --groupid com.okta.rest \ --artifactid helidon --package com.okta.rest

Slide 45

Slide 45 text

package com.okta.rest.controller; import io.helidon.security.Principal; import io.helidon.security.annotations.Authenticated; import jakarta.ws.rs.GET; import jakarta.ws.rs.Path; import jakarta.ws.rs.core.Context; @Path("/hello") public class HelloResource { @Authenticated @GET public String hello(@Context SecurityContext context) { return "Hello, " + context.userName() + "!"; } }

Slide 46

Slide 46 text

package com.okta.rest; import com.okta.rest.controller.HelloResource; import org.eclipse.microprofile.auth.LoginConfig; import jakarta.enterprise.context.ApplicationScoped; import jakarta.ws.rs.core.Application; import java.util.Set; @LoginConfig(authMethod = "MP-JWT") @ApplicationScoped public class HelloApplication extends Application { @Override public Set> getClasses() { return Set.of(HelloResource.class); } }

Slide 47

Slide 47 text

MicroProfile JWT Security mp.jwt.verify.issuer=https://dev-1337.us.auth0.com/ mp.jwt.verify.publickey.location=${mp.jwt.verify.issuer}.well- known/jwks.json https://download.eclipse.org/microprofile/microprofile-jwt-auth-2.0/microprofile-jwt-auth-spec-2.0.html io.helidon.microprofile.jwt helidon-microprofile-jwt-auth

Slide 48

Slide 48 text

Test Helidon with HTTPie https://httpie.org mvn package && java -jar target/helidon.jar http :8080/hello TOKEN=eyJraWQiOiJxOE1QMjFNNHZCVmxOSkxGbFFWNlN... http :8080/hello Authorization:"Bearer $TOKEN"

Slide 49

Slide 49 text

Verify Helidon API with HTTPie

Slide 50

Slide 50 text

@mraible Startup Performance Milliseconds 0 400 800 1200 1600 2000 Micronaut Quarkus Spring Boot Helidon 1,010 910 413 409 824 722 1,460 361 Dev Startup (gradle or mvn) Packaged Startup (java -jar)

Slide 51

Slide 51 text

@mraible What about GraphQL APIs? Why GraphQL? Does your favorite framework support GraphQL? Micronaut https://micronaut-projects.github.io/micronaut-graphql/latest/guide Quarkus https://quarkus.io/guides/smallrye-graphql Spring Boot https://spring.io/projects/spring-graphql Helidon https://helidon.io/docs/v3/#/mp/graphql

Slide 52

Slide 52 text

@mraible Secure your API with OAuth 2.1 https://aaronparecki.com/2019/12/12/21/its-time-for-oauth-2-dot-1

Slide 53

Slide 53 text

@mraible Secure your API with OAuth 2.1 https://oauth.net/2.1 PKCE is required for all clients using the authorization code flow Redirect URIs must be compared using exact string matching The Implicit grant is omitted from this specification The Resource Owner Password Credentials grant is omitted from this specification Bearer token usage omits the use of bearer tokens in the query string of URIs Refresh tokens for public clients must either be sender-constrained or one-time use

Slide 54

Slide 54 text

@mraible Authenticate with OpenID Connect (OIDC) What is OpenID Connect? Does your favorite framework support OIDC authentication? Micronaut https://guides.micronaut.io/latest/micronaut-oauth2-auth0.html Quarkus https://quarkus.io/guides/security-openid-connect-web-authentication Spring Boot https://docs.spring.io/spring-security/reference/servlet/oauth2/login Helidon https://helidon.io/docs/v3/#/mp/security/providers#OIDC-Provider

Slide 55

Slide 55 text

What about testing?

Slide 56

Slide 56 text

@mraible Build with Docker Create a Dockerfile 
 FROM openjdk:20-alpine ARG JAR_FILE=target/*.jar COPY ${JAR_FILE} app.jar EXPOSE 8080 ENTRYPOINT ["java","-jar","/app.jar"]

Slide 57

Slide 57 text

@mraible Build with Docker Build your image docker build -t . Run your image docker run -it -p 8080:8080

Slide 58

Slide 58 text

@mraible Build with Docker: Jib Get Jibby with it! mvn verify jib:build Or build directly to your Docker daemon mvn verify jib:dockerBuild https://github.com/GoogleContainerTools/jib

Slide 59

Slide 59 text

@mraible Build with Docker Micronaut uses Jib, but you must configure plugins Quarkus generates four Docker-related files Dockerfile.jvm Dockerfile.legacy-jar Dockerfile.native Dockerfile.native-micro Quarkus + Jib mvn quarkus:add-extension -Dextensions="container-image-jib"

Slide 60

Slide 60 text

@mraible Build with Docker Spring Boot 2.3+ has built-in support mvn spring-boot:build-image Uses layered JARs for faster builds dependencies snapshot-dependencies resources application https://spring.io/blog/2020/01/27/creating-docker-images-with-spring-boot-2-3-0-m1

Slide 61

Slide 61 text

@mraible Build with Docker Helidon generates three Docker-related files Dockerfile Dockerfile.jlink Dockerfile.native Helidon + Jib Not available

Slide 62

Slide 62 text

@mraible Use Micronaut CLI mn create-app ... mvn package -Dpackaging=native-image gradle nativeImage gradle dockerBuildNative Go Native with GraalVM and Micronaut https://docs.micronaut.io/latest/guide/#graal

Slide 63

Slide 63 text

@mraible Go Native with GraalVM and Quarkus Create an executable without GraalVM installed mvn package -Pnative -Dquarkus.native.container-build=true Then, build the image docker build -f src/main/docker/Dockerfile.native -t \ . And run it docker run -it -p 8080:8080 https://quarkus.io/guides/building-native-image

Slide 64

Slide 64 text

@mraible Use start.spring.io to get plugins Go Native with GraalVM and Spring Boot org.graalvm.buildtools native-maven-plugin org.springframework.boot spring-boot-maven-plugin plugins { ... id 'org.graalvm.buildtools.native' version '0.9.20' }

Slide 65

Slide 65 text

@mraible Go Native with GraalVM and Spring Boot Build the native application mvn native:compile -Pnative gradle native:compile -Pnative Build an image and Docker container mvn spring-boot:build-image -Pnative gradle bootBuildImage

Slide 66

Slide 66 text

@mraible Build the image docker build -f Dockerfile.native -t . And run it docker run --rm -p 8080:8080 Go Native with GraalVM and Helidon

Slide 67

Slide 67 text

@mraible Native Startup Performance (GraalVM 22.3) Milliseconds 0 12 24 36 48 60 June 22, 2023 33.2 47.6 39.8 25.8 23.8 33.4 Micronaut Micronaut (optimized) Quarkus Spring Boot Helidon Helidon (optimized)

Slide 68

Slide 68 text

@mraible Native Memory Used (MB) Megabytes 0 30 60 90 120 150 June 22, 2023 111 124 75 48 47 54 Micronaut Micronaut (optimized) Quarkus Spring Boot Helidon Helidon (optimized)

Slide 69

Slide 69 text

@mraible Tests Run on a 2021 MacBook Pro M1 Max

Slide 70

Slide 70 text

@mraible Demo Time! https://github.com/oktadev/auth0-java-rest-api-examples

Slide 71

Slide 71 text

Community

Slide 72

Slide 72 text

@mraible Stack Overflow Tags 0 45,000 90,000 135,000 180,000 June 21, 2023 123 140,593 3,779 1,628 Micronaut Quarkus Spring Boot Helidon

Slide 73

Slide 73 text

@mraible GitHub Stars 0 18,750 37,500 56,250 75,000 June 21, 2023 2,800 63,700 10,900 5,500 Micronaut Quarkus Spring Boot Helidon

Slide 74

Slide 74 text

https://star-history.t9t.io/#micronaut-projects/micronaut-core&quarkusio/quarkus&spring-projects/spring-boot&oracle/helidon GitHub Star Growth

Slide 75

Slide 75 text

@mraible Jobs on Indeed (DE) 0 2,500 5,000 June 21, 2023 3 2,812 342 80 Micronaut Quarkus Spring Boot Helidon

Slide 76

Slide 76 text

@mraible Twitter Followers 0 30,000 60,000 90,000 120,000 June 21, 2023 4,024 101,600 17,800 13,100 Micronaut Quarkus Spring Boot Helidon

Slide 77

Slide 77 text

@mraible JHipster Support 🤓 Spring Boot 3 - github.com/jhipster/generator-jhipster/issues/22488 Micronaut Blueprint - github.com/jhipster/generator-jhipster-micronaut - v1.0.2, 18 releases, 18 contributors, 393 commits // TODO: Micronaut 3, Reactive, Microservices, GraalVM native images Quarkus Blueprint - github.com/jhipster/generator-jhipster-quarkus - v2.0.0, 7 releases, 18 contributors, 649 commits // TODO: Quarkus 3, Dev Services, Reactive, Microservices

Slide 78

Slide 78 text

https://auth0.com/blog/full-stack-java-with-react-spring-boot-and-jhipster

Slide 79

Slide 79 text

https://developer.okta.com/blog/2020/08/17/micronaut-jhipster-heroku

Slide 80

Slide 80 text

https://developer.okta.com/blog/2021/03/08/jhipster-quarkus-oidc

Slide 81

Slide 81 text

@mraible 🏆 Quarkus provides the best DevEx, startup time, and memory usage 🚀 Micronaut is the easiest to use; startup time increased recently 🌱 Spring Boot has the strongest community, ecosystem, and growth 🔮 Helidon still has some catching up to do ⚡ Spring Boot 3 not as fast as expected My Thoughts

Slide 82

Slide 82 text

@mraible Action! New to Java? Try Spring Boot Know Spring? Trial migration paths Testing is important, invest early and often Design your apps with security in mind Use OpenID Connect and OAuth 2.1 https://unsplash.com/photos/JsTmUnHdVYQ

Slide 83

Slide 83 text

developer.auth0.com @oktadev

Slide 84

Slide 84 text

Subscribe to our Zero Index newsletter! a0.to/nl-signup/java

Slide 85

Slide 85 text

git clone https://github.com/oktadeveloper/okta-spring-web fl ux-react-example.git https://github.com/oktadev/auth0-java-rest-api-examples Use the Source, Luke!

Slide 86

Slide 86 text

Thanks! Keep in Touch raibledesigns.com @mraible Presentations speakerdeck.com/mraible Code github.com/oktadev developer.okta.com developer.auth0.com

Slide 87

Slide 87 text

developer.okta.com