org.owasp.esapi.Authenticator Maven / Gradle / Ivy
Show all versions of esapi Show documentation
/**
* OWASP Enterprise Security API (ESAPI)
*
* This file is part of the Open Web Application Security Project (OWASP)
* Enterprise Security API (ESAPI) project. For details, please see
* http://www.owasp.org/index.php/ESAPI.
*
* Copyright (c) 2007 - The OWASP Foundation
*
* The ESAPI is published by OWASP under the BSD license. You should read and accept the
* LICENSE before you use, modify, and/or redistribute this software.
*
* @author Jeff Williams Aspect Security
* @created 2007
*/
package org.owasp.esapi;
import org.owasp.esapi.errors.AuthenticationException;
import org.owasp.esapi.errors.EncryptionException;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.util.Set;
/**
* The Authenticator interface defines a set of methods for generating and
* handling account credentials and session identifiers. The goal of this
* interface is to encourage developers to protect credentials from disclosure
* to the maximum extent possible.
*
* One possible implementation relies on the use of a thread local variable to
* store the current user's identity. The application is responsible for calling
* setCurrentUser() as soon as possible after each HTTP request is received. The
* value of getCurrentUser() is used in several other places in this API. This
* eliminates the need to pass a user object to methods throughout the library.
* For example, all of the logging, access control, and exception calls need
* access to the currently logged in user.
*
* The goal is to minimize the responsibility of the developer for
* authentication. In this example, the user simply calls authenticate with the
* current request and the name of the parameters containing the username and
* password. The implementation should verify the password if necessary, create
* a session if necessary, and set the user as the current user.
*
*
* public void doPost(ServletRequest request, ServletResponse response) {
* try {
* User user = ESAPI.authenticator().login(request, response);
* // continue with authenticated user
* } catch (AuthenticationException e) {
* // handle failed authentication (it's already been logged)
* }
*
*
* @author Jeff Williams (jeff.williams .at. aspectsecurity.com) Aspect Security
* @since June 1, 2007
*/
public interface Authenticator {
/**
* Clears the current User. This allows the thread to be reused safely.
*
* This clears all threadlocal variables from the thread. This should ONLY be called after
* all possible ESAPI operations have concluded. If you clear too early, many calls will
* fail, including logging, which requires the user identity.
*/
void clearCurrent();
/**
* Calls login with the *current* request and response.
* @return Authenticated {@code User} if login is successful.
* @see HTTPUtilities#setCurrentHTTP(javax.servlet.http.HttpServletRequest, javax.servlet.http.HttpServletResponse)
*/
User login() throws AuthenticationException;
/**
* This method should be called for every HTTP request, to login the current user either from the session of HTTP
* request. This method will set the current user so that getCurrentUser() will work properly.
*
* Authenticates the user's credentials from the HttpServletRequest if
* necessary, creates a session if necessary, and sets the user as the
* current user.
*
* Specification: The implementation should do the following:
* 1) Check if the User is already stored in the session
* a. If so, check that session absolute and inactivity timeout have not expired
* b. Step 2 may not be required if 1a has been satisfied
* 2) Verify User credentials
* a. It is recommended that you use
* loginWithUsernameAndPassword(HttpServletRequest, HttpServletResponse) to verify credentials
* 3) Set the last host of the User (ex. user.setLastHostAddress(address) )
* 4) Verify that the request is secure (ex. over SSL)
* 5) Verify the User account is allowed to be logged in
* a. Verify the User is not disabled, expired or locked
* 6) Assign User to session variable
*
* @param request
* the current HTTP request
* @param response
* the HTTP response
*
* @return
* the User
*
* @throws AuthenticationException
* if the credentials are not verified, or if the account is disabled, locked, expired, or timed out
*/
User login(HttpServletRequest request, HttpServletResponse response) throws AuthenticationException;
/**
* Verify that the supplied password matches the password for this user. Password should
* be stored as a hash. By default, this method verifies password hashes created via the
* {@code hashPassword(password, accountName)} method in this class, however see WARNING
* the {@code hashPassword} method.
*
* This method is typically used for "reauthentication" for the most sensitive functions, such
* as transactions, changing email address, and changing other account information.
*
* @param user
* the user who requires verification
* @param password
* the hashed user-supplied password
*
* @return
* true, if the password is correct for the specified user
*
* @see #hashPassword(String password, String accountName)
*/
boolean verifyPassword(User user, String password);
/**
* Logs out the current user.
*
* This is usually done by calling User.logout on the current User.
*/
void logout();
/**
* Creates a new User with the information provided. Implementations should check
* accountName and password for proper format and strength against brute force
* attacks ( verifyAccountNameStrength(String), verifyPasswordStrength(String, String) ).
*
* Two copies of the new password are required to encourage user interface designers to
* include a "re-type password" field in their forms. Implementations should verify that
* both are the same.
*
* WARNING: The implementation of this method as defined in the
* default reference implementation class, {@code FileBasedAuthenticator},
* uses a password hash algorthim that is known to be weak. You are advised
* to replace the default reference implementation class with your own custom
* implementation that uses a stronger password hashing algorithm.
* See class comments in * {@code FileBasedAuthenticator} for further details.
*
* @param accountName
* the account name of the new user
* @param password1
* the password of the new user
* @param password2
* the password of the new user. This field is to encourage user interface designers to include two password fields in their forms.
*
* @return
* the User that has been created
*
* @throws AuthenticationException
* if user creation fails due to any of the qualifications listed in this method's description
*/
User createUser(String accountName, String password1, String password2) throws AuthenticationException;
/**
* Generate a strong password. Implementations should use a large character set that does not
* include confusing characters, such as i I 1 l 0 o and O. There are many algorithms to
* generate strong memorable passwords that have been studied in the past.
*
* @return
* a password with strong password strength
*/
String generateStrongPassword();
/**
* Generate strong password that takes into account the user's information and old password. Implementations
* should verify that the new password does not include information such as the username, fragments of the
* old password, and other information that could be used to weaken the strength of the password.
*
* @param user
* the user whose information to use when generating password
* @param oldPassword
* the old password to use when verifying strength of new password. The new password may be checked for fragments of oldPassword.
*
* @return
* a password with strong password strength
*/
String generateStrongPassword(User user, String oldPassword);
/**
* Changes the password for the specified user. This requires the current password, as well as
* the password to replace it with. The new password should be checked against old hashes to be sure the new password does not closely resemble or equal any recent passwords for that User.
* Password strength should also be verified. This new password must be repeated to ensure that the user has typed it in correctly.
*
* @param user
* the user to change the password for
* @param currentPassword
* the current password for the specified user
* @param newPassword
* the new password to use
* @param newPassword2
* a verification copy of the new password
*
* @throws AuthenticationException
* if any errors occur
*/
void changePassword(User user, String currentPassword, String newPassword, String newPassword2) throws AuthenticationException;
/**
* Returns the User matching the provided accountId. If the accoundId is not found, an Anonymous
* User or null may be returned.
*
* @param accountId
* the account id
*
* @return
* the matching User object, or the Anonymous User if no match exists
*/
User getUser(long accountId);
/**
* Returns the User matching the provided accountName. If the accoundId is not found, an Anonymous
* User or null may be returned.
*
* @param accountName
* the account name
*
* @return
* the matching User object, or the Anonymous User if no match exists
*/
User getUser(String accountName);
/**
* Gets a collection containing all the existing user names.
*
* @return
* a set of all user names
*/
Set getUserNames();
/**
* Returns the currently logged in User.
*
* @return
* the matching User object, or the Anonymous User if no match
* exists
*/
User getCurrentUser();
/**
* Sets the currently logged in User.
*
* @param user
* the user to set as the current user
*/
void setCurrentUser(User user);
/**
* Returns a string representation of the hashed password, using the
* accountName as the salt. The salt helps to prevent against "rainbow"
* table attacks where the attacker pre-calculates hashes for known strings.
* This method specifies the use of the user's account name as the "salt"
* value. The Encryptor.hash method can be used if a different salt is
* required.
*
* WARNING: The implementation of this method as defined in the
* default reference implementation class, {@code FileBasedAuthenticator},
* is know to be extremely weak. The reference implementation class was
* meant to be an example implementation and generally should be avoided
* and replaced with your own implementation. See class comments in
* {@code FileBasedAuthenticator} for further details.
*
* @param password
* the password to hash
* @param accountName
* the account name to use as the salt
*
* @return
* the hashed password
* @throws EncryptionException
*
* @see org.owasp.esapi.reference.FileBasedAuthenticator FileBasedAuthenticator,
* the default reference implementation of this interface.
*/
String hashPassword(String password, String accountName) throws EncryptionException;
/**
* Removes the account of the specified accountName.
*
* @param accountName
* the account name to remove
*
* @throws AuthenticationException
* the authentication exception if user does not exist
*/
void removeUser(String accountName) throws AuthenticationException;
/**
* Ensures that the account name passes site-specific complexity requirements, like minimum length.
*
* @param accountName
* the account name
*
* @throws AuthenticationException
* if account name does not meet complexity requirements
*/
void verifyAccountNameStrength(String accountName) throws AuthenticationException;
/**
* Ensures that the password meets site-specific complexity requirements, like length or number
* of character sets. This method takes the old password so that the algorithm can analyze the
* new password to see if it is too similar to the old password. Note that this has to be
* invoked when the user has entered the old password, as the list of old
* credentials stored by ESAPI is all hashed.
* Additionally, the user object is taken in order to verify the password and account name differ.
*
* @param oldPassword
* the old password
* @param newPassword
* the new password
* @param user
* the user
*
* @throws AuthenticationException
* if newPassword is too similar to oldPassword or if newPassword does not meet complexity requirements
*/
void verifyPasswordStrength(String oldPassword, String newPassword, User user) throws AuthenticationException;
/**
* Determine if the account exists.
*
* @param accountName
* the account name
*
* @return true, if the account exists
*/
boolean exists(String accountName);
}