Modern applications rely on user authentication, but it can present Java developers with a difficult challenge, as well as a range of framework-specific options to choose from. We have seen many Spring developers start with a simple, home-grown authentication service they plan to replace “later” with a more robust option… only for that homegrown service to bikeshed its way to a permanent place in the stack. To end this cycle of heartbreak, this post will show how simple it is to implement an enterprise-grade auth service, even in a simple app.
In this tutorial, you’ll create an application that displays user information. You’ll configure it manually first to see its drawbacks. Then, we’ll use a more professional approach. By the end of this tutorial, you’ll have a Spring-based Java application that uses OAuth 2.0 to authenticate users, and it will take you 5 minutes to make these changes!
Create Your Java Application with Spring
Let’s start by creating the project structure. You’ll use Spring Initializer to create the application. Go to start.spring.io and fill in the following information:
- Project: Maven Project
- Language: Java
- Group: com.okta.authorizationapp
- Artifact: oauth
- Dependencies:
- Spring Web
- Spring Security
- Thymeleaf
You can also generate the project from the command line. Paste the following command in your terminal to download the project with the same configuration as above:
curl https://start.spring.io/starter.zip \
-d dependencies=web,thymeleaf,security \
-d packageName=com.okta.authorizationapp \
-d name=authorization-app \
-d type=maven-project \
-o java-authorization-app.zip
That’s it! Now your Java project structure is created, and you can start developing your app.
Build User Security on Your Own
This tutorial will use Maven, but you can easily do it using Gradle if you prefer.
First, import the project in your favorite IDE/editor. Right now, your project has only one class, AuthorizationAppApplication
that bootstraps the application. When you run this class, the server starts, and you can go to your browser to see the results.
However, you first need a page to access, so let’s create a home page.
Inside src/main/java/com/okta/authorizationapp/controller/
create the class HomeController
:
@Controller
public class HomeController {
private Map<String, LocalDateTime> usersLastAccess = new HashMap<>();
@GetMapping("/")
public String getCurrentUser(@AuthenticationPrincipal User user, Model model) {
String username = user.getUsername();
model.addAttribute("username", username);
model.addAttribute("lastAccess", usersLastAccess.get(username));
usersLastAccess.put(username, LocalDateTime.now());
return "home";
}
}
This class defines a controller for the /
path. When you access your app without defining any other path, this code will execute.
The controller’s first important action retrieves the current user’s information. Since you annotated your user
attribute with AuthenticationPrincipal
, Spring Security will automatically retrieve this information.
The controller also receives a model
parameter that stores the data used to render the page. Right now, this data is the username
and the last time the user accessed your application.
Create Dynamic Messages on User Login
The final step is to update the user’s last access date and define which HTML template should render the request. In your case, the endpoint is called home
. Spring will search for a home.html
file inside the src/main/resources/templates
folder.
You don’t have this file yet, so let’s go there and create it:
<html>
<head>
<title>Java OAuth 2.0 Tutorial - Homepage</title>
</head>
<body>
<h1 th:text="'Welcome, ' + ${username} + '!'"></h1>
<ul>
<li th:if="${lastAccess}" th:text="'Last access: ' + $"></li>
</ul>
</body>
</html>
This is an HTML file altered slightly by Thymeleaf, one of the libraries you imported when you created the project. Thymeleaf receives the model object from the server and renders the values from it in HTML. Just type ${variable}
to refer to a variable in the model
object.
The th:text
attribute will let you define a dynamic text in the HTML element. Here, we use it to display a dynamic greeting, and the last time the user accessed the application.
The first time a user accesses your app, there is no prior access recorded. To make sure you’re not presenting a meaningless field, use th:if
. If the field is null
, the li
tag is not rendered, and the user won’t see it.
Add Basic Login to Your Java Spring App
Now you have the endpoint, you just need to add security to your app.
Inside src/main/javacom/okta/authorizationapp/configuration/
create the class SecurityConfiguration
:
@EnableWebSecurity
public class SecurityConfiguration extends WebSecurityConfigurerAdapter {
private PasswordEncoder passwordEncoder = new BCryptPasswordEncoder();
@Override
protected void configure(AuthenticationManagerBuilder auth) throws Exception {
auth.inMemoryAuthentication()
.passwordEncoder(passwordEncoder())
.withUser("john.doe")
.password(passwordEncoder().encode("secret"))
.roles("USER");
}
@Bean
public PasswordEncoder passwordEncoder() {
return passwordEncoder;
}
}
This class will ensure that users must log in to access your application. Right now there is only one user named john.doe
who can log into the app.
Run the application by calling the main
method inside AuthorizationAppApplication
. You can also run it from the command line. Inside the project folder, run the following command:
mvn spring-boot:run
When you go to http://localhost:8080
you should see the following login page:
Type john.doe
and secret
as username and password. You should be redirected to the home page. On the first visit, only Welcome, john.doe!
will display. From the second visit, you should also see the last access:
You now have an application that manages security. Good job!
There is a big problem, though… Right now, you can only login with one user. Even worse, the user information is hardcoded in your app. To simplify user access and security, you can use Okta to manage your authentication. It will provide you a very simple way to integrate with OAuth 2.0 in less than 5 minutes. Let’s configure OAuth 2.0 in your sample app to see how easy it is. Let’s start by creating an Okta account.
Create an Okta Account
If you don’t have an Okta account, go ahead and create one. Once you have signed up, go through the following steps:
- Login in to your account
- Click on Applications > Add Application
You will Fokbe redirected to the following page:
- Select Web and click Next
Fill in the following options in the form:
- Name:
hello-world
- Base URIs:
http://localhost:8080
- Login redirect URLs:
http://localhost:8080/login/oauth2/code/okta
- Grant Type allowed:
- Client Credentials
- Authorization Code
- Click Done.
Now you can use your Okta application to authenticate users to your app.
Use OAuth 2.0: A Fast, Professional Approach
Let’s start by adding Okta’s library to your project.
Go to the pom.xml
and add Okta’s Spring Boot starter:
<dependency>
<groupId>com.okta.spring</groupId>
<artifactId>okta-spring-boot-starter</artifactId>
<version>1.3.0</version>
</dependency>
Okta will manage your app authentication, so you can delete the SecurityConfiguration
class.
Inside HomeController
, make the following changes:
@GetMapping("/")
public String getCurrentUser(@AuthenticationPrincipal OidcUser user, Model model) {
String email = user.getEmail();
model.addAttribute("email", email);
model.addAttribute("lastAccess", usersLastAccess.get(email));
model.addAttribute("firstName", user.getGivenName());
model.addAttribute("lastName", user.getFamilyName());
usersLastAccess.put(email, LocalDateTime.now());
return "home";
}
Your endpoint will now receive an OidcUser
compatible with OAuth 2.0. This class provides much more user information than you had before, so you can modify your HTML to display it. Replace username
with email
, and add firstName
, and lastName
, which are fields you didn’t have before. To do this, go to the hello.html
and make the following changes:
<body>
<h1 th:text="'Welcome, ' + ${email} + '!'"></h1>
<ul>
<li th:if="${lastAccess}" th:text="'Last access: ' + $"></li>
<li th:if="${firstName}" th:text="'First name: ' + $"></li>
<li th:if="${lastName}" th:text="'Last name: ' + $"></li>
</ul>
</body>
You are still greeting the user as before, but you’re also displaying the new information from the endpoint. You have made all the code changes and now just need to add some configurations. Create an okta.env
file in the root directory of your app with the following environment variables.
export OKTA_OAUTH2_ISSUER=/oauth2/default
export OKTA_OAUTH2_CLIENT_ID={CLIENT_ID}
export OKTA_OAUTH2_CLIENT_SECRET={CLIENT_SECRET}
You will find {CLIENT_ID}
and {CLIENT_SECRET}
in the application’s page from the Okta dashboard. To access it, follow the steps below:
- In your Okta dashboard, go to Applications
- Select the hello-world application
- Click on the General tab
You should see both values in the Client Credentials area. Your {yourOktaDomain}
will be visible in your Okta dashboard, just click on Dashboard in the menu. You will see the Org URL in the right upper corner.
Once you’ve pasted these values into okta.env
, run the following commands to start your app.
source okta.env
mvn spring-boot:run
That’s it!
Log Into Your Spring App With OAuth 2.0 Enabled
Navigate to http://localhost:8080
. Your app will redirect you to Okta’s login page:
After logging in, you’ll be redirected to your application and see a message like this:
You’ve done it! In 5 minutes you added OAuth 2.0 in your application with very little configuration along the way.
Learn More About Spring Security, Spring Boot and Java Authentication
If you want to take a look at the completed source code, you can access it on GitHub.
Do you want to read more about OAuth 2.0 and Java in general? You might be interested in the following articles:
- A Quick Guide to Spring Boot Login Options
- Spring Method Security with PreAuthorize
- Monitor Your Java Apps with Spring Boot Actuator
For more articles like this one, follow @oktadev on Twitter. We also regularly publish screencasts to our YouTube channel.
Top comments (0)