19.4. Login and Registration Forms

With a User class in place, we can now create controllers and views for creating a user and verifying their credentials.

19.4.1. Creating AuthenticationController

In the controllers package, create a new class named AuthenticationController. Since this controller will deal with User objects, it needs a UserRepository instance.

21
22
23
24
25
26
27
@Controller
public class AuthenticationController {

   @Autowired
   UserRepository userRepository;

}

19.4.2. Session-Handling Utilities

Before creating handler methods for rendering and processing our login and registration forms, we need some utility methods for working with sessions. Below the definition of userRepository, let’s add the following class members:

27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
private static final String userSessionKey = "user";

public User getUserFromSession(HttpSession session) {
   Integer userId = (Integer) session.getAttribute(userSessionKey);
   if (userId == null) {
      return null;
   }

   Optional<User> user = userRepository.findById(userId);

   if (user.isEmpty()) {
      return null;
   }

   return user.get();
}

private static void setUserInSession(HttpSession session, User user) {
   session.setAttribute(userSessionKey, user.getId());
}

This code allows us to store and retrieve the login status of a user in a session. More specifically, a logged-in user’s user ID will be stored in their session.

Sample session data for a logged-in user

session_id

81LfWG9

user

42

The static field userSessionKey is the key used to store user IDs. setUserInSession uses an HttpSession object (part of the standard javax.servlet.http package) to store key/value pair. getUserFromSession looks for data with the key user in the user’s session. If it finds one, it attempts to retrieve the corresponding User object from the database. If no user ID is in the session, or if there is no user with the given ID, null is returned.

Note

The HttpSession class handles the details of session creation and lookup for us, including generating unique session IDs and session cookies.

These utility methods will allow our handlers to manage authentication.

19.4.3. Form DTOs

Our login and registration forms will use DTOs to help with form rendering and processing. Furthermore, since these forms will be similar—both require a username and password—we’ll use inheritance in creating our DTOs.

The DTO for the login form needs only username and password fields.

10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
public class LoginFormDTO {

   @NotNull
   @NotBlank
   @Size(min = 3, max = 20, message = "Invalid username. Must be between 3 and 20 characters.")
   private String username;

   @NotNull
   @NotBlank
   @Size(min = 5, max = 30, message = "Invalid password. Must be between 5 and 30 characters.")
   private String password;

   public String getUsername() {
      return username;
   }

   public void setUsername(String username) {
      this.username = username;
   }

   public String getPassword() {
      return password;
   }

   public void setPassword(String password) {
      this.password = password;
   }

}

LoginFormDTO should live in the models.dto package.

Tip

To better understand this approach, think of a DTO associated with a form as an object that represents each of the form fields. Using a DTO to represent the data associated with a form makes form rendering and processing much easier when the form does not line up with a specific model class.

Note

In the class above, we have a password field that will store a plain-text password. However, this does not contradict our early imperative about NOT storing passwords, since LoginFormDTO is not a persistent class.

Our registration form will ask for a username/password pair, and then ask the user to confirm the password by typing it in again. So the associated DTO can extend LoginFormDTO and add an additional field for password verification.

 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
public class RegisterFormDTO extends LoginFormDTO {

   private String verifyPassword;

   public String getVerifyPassword() {
      return verifyPassword;
   }

   public void setVerifyPassword(String verifyPassword) {
      this.verifyPassword = verifyPassword;
   }

}

RegisterFormDTO should also live in the models.dto package.

19.4.4. The Registration Form

We are now ready to build our form handlers. Before we can authenticate a user, we must have users in the application, so we’ll build the registration form first.

To render the form within AuthenticationController is simple:

48
49
50
51
52
53
@GetMapping("/register")
public String displayRegistrationForm(Model model) {
   model.addAttribute(new RegisterFormDTO());
   model.addAttribute("title", "Register");
   return "register";
}

Note

When passing an object into the view with model.addAttribute, specifying a label for the object is optional. If a label is not specified the class name is used, with the first letter converted to lowercase.

In the method above, model.addAttribute(new RegisterFormDTO()) will pass a RegisterFormDTO object in with the label registerFormDTO.

The registration form (in templates/register.html) uses the three DTO fields to render the form fields:

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
<!DOCTYPE html>
<html lang="en" xmlns:th="http://www.thymeleaf.org/">
<head th:replace="fragments :: head"></head>
<body class="container">

<header th:replace="fragments :: header"></header>

<form method="post">
   <div class="form-group">
      <label>Username
            <input class="form-control" th:field="${registerFormDTO.username}" />
      </label>
      <p class="error" th:errors="${registerFormDTO.username}"></p>
   </div>
   <div class="form-group">
      <label>Password
            <input class="form-control" th:field="${registerFormDTO.password}" type="password" />
      </label>
      <p class="error" th:errors="${registerFormDTO.password}"></p>
   </div>
   <div class="form-group">
      <label>Verify Password
            <input class="form-control" th:field="${registerFormDTO.verifyPassword}" type="password" />
      </label>
   </div>

   <input type="submit" class="btn btn-primary" value="Register" />
</form>

</body>
</html>

The form processing handler is more complicated. Let’s look at it, and then break it down in detail.

55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
@PostMapping("/register")
public String processRegistrationForm(@ModelAttribute @Valid RegisterFormDTO registerFormDTO,
                     Errors errors, HttpServletRequest request,
                     Model model) {

   if (errors.hasErrors()) {
      model.addAttribute("title", "Register");
      return "register";
   }

   User existingUser = userRepository.findByUsername(registerFormDTO.getUsername());

   if (existingUser != null) {
      errors.rejectValue("username", "username.alreadyexists", "A user with that username already exists");
      model.addAttribute("title", "Register");
      return "register";
   }

   String password = registerFormDTO.getPassword();
   String verifyPassword = registerFormDTO.getVerifyPassword();
   if (!password.equals(verifyPassword)) {
      errors.rejectValue("password", "passwords.mismatch", "Passwords do not match");
      model.addAttribute("title", "Register");
      return "register";
   }

   User newUser = new User(registerFormDTO.getUsername(), registerFormDTO.getPassword());
   userRepository.save(newUser);
   setUserInSession(request.getSession(), newUser);

   return "redirect:";
}
  • Lines 55-58: Define the handler method at the route /register that takes a valid RegisterFormDTO object, associated errors, and a Model. In addition, the method needs an HttpServletRequest object. This object represents the incoming request, and will be provided by Spring.

  • Lines 60-63: Return the user to the form if an validation errors occur.

  • Line 65: Retrieve the user with the given username from the database.

  • Lines 67-71: If a user with the given username already exists, register a custom error with the errors object and return the user to the form. See the note on using errors.rejectValue below.

  • Lines 73-79: Compare the two passwords submitted. If they do not match, register a custom error and return the user to the form.

  • Lines 81-83: At this point, we know that a user with the given username does NOT already exist, and the rest of the form data is valid. So we create a new user object, store it in the database, and then create a new session for the user.

  • Line 85: Finally, redirect the user to the home page.

Note

The Errors class we have been using in conjunction with model binding will always contain information about errors related to validation annotations on the given model. However, it can also be used to manually generate additional errors. In the method above, we call:

errors.rejectValue("username", "username.alreadyexists",
                   "A user with that username already exists");

errors.rejectValue takes three parameters:

  1. The field containing the error.

  2. A label representing the error. This allows error messages to be imported from another file. While we don’t have such a file, this parameter is required.

  3. A default message to use if no external error message file is available (as is the case here).

This is a good time to test your application. Start it up, navigate to /register and try to create a user. If everything goes well, you will see a new row in the user table of the database.

The user table with a new row containing id, pw_hash, and username values

A new row in the user table

19.4.5. The Login Form

Rendering the login form is similar to rendering the registration form:

89
90
91
92
93
94
@GetMapping("/login")
public String displayLoginForm(Model model) {
   model.addAttribute(new LoginFormDTO());
   model.addAttribute("title", "Log In");
   return "login";
}

The form template itself should be placed in templates/login.html, and is also similar to the registration template:

<!DOCTYPE html>
<html lang="en" xmlns:th="http://www.thymeleaf.org/">
<head th:replace="fragments :: head"></head>
<body class="container">

<header th:replace="fragments :: header"></header>

<form method="post">
   <div class="form-group">
      <label th:for="username">Username
            <input class="form-control" th:field="${loginFormDTO.username}">
      </label>
      <p class="error" th:errors="${loginFormDTO.username}"></p>
   </div>
   <div class="form-group">
      <label>Password
            <input class="form-control" th:field="${loginFormDTO.password}" type="password">
      </label>
      <p class="error" th:errors="${loginFormDTO.password}"></p>
   </div>

   <input type="submit" class="btn btn-primary" value="Log In" />
</form>

<p>Don't have an account? <a href="/register">Register for one.</a></p>

</body>
</html>

As usual, processing the form is more complicated. Again, we’ll break it down in detail.

 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
@PostMapping("/login")
public String processLoginForm(@ModelAttribute @Valid LoginFormDTO loginFormDTO,
                  Errors errors, HttpServletRequest request,
                  Model model) {

   if (errors.hasErrors()) {
      model.addAttribute("title", "Log In");
      return "login";
   }

   User theUser = userRepository.findByUsername(loginFormDTO.getUsername());

     if (theUser == null) {
         errors.rejectValue("username", "user.invalid", "The given username does not exist");
         model.addAttribute("title", "Log In");
         return "login";
     }

     String password = loginFormDTO.getPassword();

     if (!theUser.isMatchingPassword(password)) {
      errors.rejectValue("password", "password.invalid", "Invalid password");
      model.addAttribute("title", "Log In");
      return "login";
   }

   setUserInSession(request.getSession(), theUser);

   return "redirect:";
}
  • Lines 96-104: The method definition, parameters, and initial error checking are nearly identical to the registration POST handler above.

  • Line 106: Retrieves the User object with the given password from the database.

  • Lines 108-112: If no such user exists, register a custom error and return to the form.

  • Line 114: Retrieves the submitted password from the form DTO.

  • Lines 116-120: If the password is incorrect, register a custom error and return to the form. Password verification uses the User.isMatchingPassword() method, which handles the details associated with checking hashed passwords.

  • Line 122: At this point, we know the given user exists and that the submitted password is correct. So we create a new session for the user.

  • Line 124: Finally, redirect the user to the home page.

Now you can test your login form. Upon successful form submission, you should be redirected to the home page. To verify that a session was created, open Firefox’s developer tools and navigate to the Storage pane. Select Cookies > http://localhost:8080 in the left-hand pane and you should see a cookie with the key JSESSIONID. This is the session cookie created by the application. (You may see other cookies as well, which is okay.)

A session cookie visible in the Storage pane of the browser's dev tools.

A session cookie for our application

19.4.6. Logging Out

After the complicated processes of user registration and login, logging a user out is refreshingly simple.

127
128
129
130
131
@GetMapping("/logout")
public String logout(HttpServletRequest request){
   request.getSession().invalidate();
   return "redirect:/login";
}

To log out, we simply invalidate the session associated with the given user. This removes all data from the session, so that when the user makes a subsequent request, they will be forced to log in again.

Note

The code for this section is available in the login-reg-forms branch of the coding-events-demo repository.

19.4.7. Check Your Understanding

Question

What is the name of the new method we have introduced on the Errors object?

  1. Errors.hasErrors()

  2. Errors.errors()

  3. Errors.isNotEmpty()

  4. Errors.rejectValue()

Question

Which developer tool panel can we use to verify that a user session has been started?

  1. Console

  2. Network

  3. Storage

  4. Performance