Skip to content

This is a web-app client, that written on top of Spring Security, and uses the "authentication-flows" JAR. The Authentication-Flows JAR implements all authentication flows: create account, forgot password, change password by user request, force change password if password is expired, locks the accont after pre-configured login failures.

Notifications You must be signed in to change notification settings

OhadR/Authentication-Flows-client

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

Authentication-Flows-Client Build Status

This project is a web-app client, that writen on top of Spring Security, and uses the "authentication-flows" JAR. The "authentication-flows" project is here (under 'oAuth sample' repository).

23-02-2016: Spring Versions Updated

On 23-02-2016, we have updated the authentication flows dependencies to Spring newest!

  • spring security version = 4.0.3.RELEASE
  • spring version = 4.2.4.RELEASE

In addition, we have changed the build tool from Maven to Gradle. If you wish to use the older version, i.e Maven and older Spring versions (3.1.X), you can find it on a separated branch.

The Authentication-Flows has its own dependencies; read about it in its own README.

The Authentication-Flows JAR implements all authentication flows:

  • create account,
  • forgot password,
  • change password by user request,
  • force change password if password is expired,
  • locks the accont after pre-configured login failures.

NOTE: Spring's generic login form is not good enought, since it lacks links as "register" and "forgot password". So the client has its own login page, as well as other required forms. These forms can be found in this project The client, which is a WAR, includes the HTMLs and JSPs, under webapp/login directory. These html and JSPs forms submits to a URL that the authentication-flows JAR responds to.

Configuration

  1. Client's Spring-Beans.XML

1.1. Component-Scan

the XML should contain to the component-scan path the following paths:

com.ohadr.auth_flows.*
com.ohadr.crypto.*

1.2. password encoder

add bean in the spring XML. it is in use in the UserActionController.

	<sec:authentication-manager alias="authenticationManager">
		<sec:authentication-provider user-service-ref="userDetailsService" >
			<sec:password-encoder hash="sha-256">
				<sec:salt-source user-property="username"/>
			</sec:password-encoder>
		</sec:authentication-provider>
	</sec:authentication-manager>

	<bean id="passwordEncoder" 	class="org.springframework.security.authentication.encoding.ShaPasswordEncoder">
		<constructor-arg value="256"/>
	</bean>
	<sec:form-login 
			login-page="/login/login.htm" 
			authentication-success-handler-ref="authenticationSuccessHandler"
			authentication-failure-handler-ref="authenticationFailureHandler" />

1.3. authentication success handler

add this to the <form-login> block:

	authentication-success-handler-ref="authenticationSuccessHandler"

after a successful login, we need to check whether the user has to change hos password (if it is expired).

1.4. authentication failure handler

add this to the <form-login> block:

	authentication-failure-handler-ref="authenticationFailureHandler"

and this bean:

	<bean id="authenticationFailureHandler" class="com.ohadr.auth_flows.core.AuthenticationFailureHandler">
		<constructor-arg value="/login/login.htm?login_error=1"/>
		<property name="accountLockedUrl" value="/login/accountLocked.htm" />
	</bean>

1.5. velocity - for better emails...

issue OhadR/oAuth2-sample#31 : read content of emails from a file. For this, we use velocity.

    <bean id="velocityEngine" class="org.springframework.ui.velocity.VelocityEngineFactoryBean">
        <property name="velocityProperties">
            <value>
                resource.loader=class
                class.resource.loader.class=org.apache.velocity.runtime.resource.loader.ClasspathResourceLoader
            </value>
        </property>
    </bean>
  1. Database

see https://github.com/OhadR/oAuth2-sample/tree/master/authentication-flows#2-database

  1. UI Forms

As stated beofre, the cliend application is responsible for the html/jsp forms, that passes the data to the authentication-flows module. It makes sence, because every client my have its own UI, its own logos and its own look, so there is no sense that the backend module supplies the UI forms.

3.1. login.htm

3.2. createAccount.jsp

3.3. forgotPassword.htm

3.4. setNewPassword.jsp

3.5. changePassword.jsp - TBD

3.6. accountLocked.htm

In DaoAuthenticationProvider.additionalAuthenticationChecks(), Spring checks the password the user entered, in front of the one in the DB. It calls to passwordEncoder.isPasswordValid(). IT gets there only AFTER the check that user exists in 'user' table, and in 'authorities' table.

Since we use Spring's 'username' column name in the DB (and not 'email' as planned), it is important to know that we expect the username to be the email of the user. we count on it by sending email to this address.

upon login, Spring makes the checks of creds, and some checks before and some after. before: AbstractUserDetailsAuthenticationProvider.authenticate(), there are the checks. then there is a call to preAuthenticationChecks.check(user). There, Spring checks the account is non-locked, enabled, and non-expired. after: Spring checks that the account credentials are non-expired. this is great for implementations that forces the user to change his password once in a while.

IMPORTANT: in Spring, once the account is locked, or disabled, etc - the "Authentication Failure Listener" will not be called! (in my old algorith, I handled the 'locks', so as far as Spring concerned, the account was never locked, or password was never expired.

  1. properties

com.ohadr.auth-flows.linksExpirationMinutes -

com.ohadr.auth-flows.mail.username - the username in the SMTP server. E.g. if you use Google account: [email protected]

com.ohadr.auth-flows.mail.password - the password in the SMTP server.

com.ohadr.auth-flows.email.fromField - the string that appears in the "from" field in the emails that are sent to users. E.g 'App Admin'.

com.ohadr.auth-flows.isREST - 'true' if this client-app uses authentication-flows' REST APIs; 'false' o/w.

com.ohadr.auth-flows.endpoints.accountActivatedEndpointUrl - Basically this is the URL that the controller redirects to, upon account activation (meaning, when the user clicks the link in the verification email that he gets after he subscribes). This property lets the auth-flows to be generic in sense that it does not force the client-app to a specific activation endpoint. So each client-app can has its own endpoint URL. For example, one app can have this endpoint: /login/accountActivated.htm while another app can use this one: /activatation.jsp.

Notes

Authentication-Flows on Maven Repository

If you do not want to download the source from GitHub (which is recommended), you can use it directly from Maven Central Repository. Add this dependency to your POM.xml:

<dependency>
  <groupId>com.ohadr</groupId>
  <artifactId>authentication-flows</artifactId>
  <version>1.1.4</version>
</dependency>

Note the version - make sure you use the latest.

Redirect URI

"Redirect URI" has a big role in authentication, as it lets the server know where to redirect the user to. The redirect URI can be transferred in the registration email, for example so after the registration is finished and account is activated, the server still knows where to redirect the client to. In this example, I did not support this, in order to simplify the solution.

Remember-Me

the "remember me" feature is not implemented here, since it does not serve the purpose of the authentication flows. It is possible to read from the "policy" table that value that indicates for how long the "remember me" cookie will be valid, but it is up to the developer to decide whether to implement it or not. I assume the user knows how to use Spring's Remember-Me feature, otherwise read the documentations. But for a short summary:

  • in the client's beans.xml add the remember me tag:
	<security:remember-me 
			data-source-ref="dataSource"
			user-service-ref="userDetailsService"/>
  • in the UserActionController.java, uncomment the lines:
	
	@Autowired
	private AbstractRememberMeServices rememberMeService;
  • then, in UserActionController.java,
	//read the value from the policy (from the DB):
	int rememberMeTokenValidityInDays = settings.getRememberMeTokenValidityInDays();

	//get the "remem-me" bean and update its validity:
	rememberMeService.setTokenValiditySeconds(rememberMeTokenValidityInDays * 60 * 60 * 24);

and you are ready to go.

Why "Secret Question" mechanism is a Bad Thing

The logic of "Secret Question" escapes me. Since the dawn of computer security we have been telling people, "DON'T make a password that is information about yourself that a hacker could discover or guess, like the name of your high school, or your favorite color. A hacker might be able to look up the name of your high school, or even if they don't know you or know anything about you, if you still live near where you went to school they might get it by tryinging local schools until they hit it. There are a small number of likely favorite colors so a hacker could guess that. Etc. Instead, a password should be a meaningless combination of letters, digits, and punctuation." But now we also tell them, "But! If you have a difficult time remembering that meaningless combination of letters, digits, and punctuation, no problem! Take some information about yourself that you can easily remember -- like the name of your high school, or your favorite color -- and you can use that as the answer to a 'security question', that is, as an alternative password."

Indeed, security questions make it even easier for the hacker than if you just chose a bad password to begin with. At least if you just used a piece of personal information for your password, a hacker wouldn't necessarily know what piece of personal information you used. Did you use the name of your dog? Your birth date? Your favorite ice cream flavor? He'd have to try all of them. But with security questions, we tell the hacker exactly what piece of personal information you used as a password!

Instead of using security questions, why don't we just say, "In case you forget your password, it is displayed on the bottom of the screen. If you're trying to hack in to someone else's account, you are absolutely forbidden from scrolling down." It would be only slightly less secure. source

Why should we NEVER use CAPTCHA

Well, here is why.

Bitdeli Badge

About

This is a web-app client, that written on top of Spring Security, and uses the "authentication-flows" JAR. The Authentication-Flows JAR implements all authentication flows: create account, forgot password, change password by user request, force change password if password is expired, locks the accont after pre-configured login failures.

Resources

Stars

Watchers

Forks

Packages

No packages published

Contributors 3

  •  
  •  
  •