Search code examples
springapachespring-securityoneloginshibboleth-sp

Java - Spring security, Shibboleth (apache) and onelogin


The actual Spring Security configuration is like this:

@Configuration
@EnableWebSecurity
public class WebSecurityConfig extends WebSecurityConfigurerAdapter {

    @Override
    protected void configure(HttpSecurity http) throws Exception {
        http.authorizeRequests()
        .antMatchers("/uri1/**").hasAuthority(Permission.AUTHORITY1.toString())
        .antMatchers("/uri2/**").hasAuthority(Permission.AUTHORITY2.toString())
        .anyRequest().hasAuthority(Permission.AUTHORITY3.toString())
        .and().httpBasic()
        .realmName("App").and().csrf().disable();
        http.authorizeRequests();
        http.headers().frameOptions().sameOrigin().cacheControl().disable();
    }
    
    @Bean
    public Filter shallowEtagHeaderFilter() {
        return new ShallowEtagHeaderFilter();
    }
}

And the web MVC configuration is like this:

@Configuration
public class DefaultView extends WebMvcConfigurerAdapter{

    @Override
    public void addViewControllers( ViewControllerRegistry registry ) {
        registry.addViewController( "/" ).setViewName( "forward:myPage.html" );
        registry.setOrder( Ordered.HIGHEST_PRECEDENCE);
        super.addViewControllers( registry );
    }
}

I have to replace the httpBasic authentification done in Spring Security by an authentification using onelogin (so with SAML if I understood what I found on the Internet).

By doing research, I found that a possibility was to use Shibboleth on the Apache server and an other was to use a plugin in Spring Security to manage SAML.

For the first solution (shibboleth), the aim is to manage onelogin authentification directly on Apache server (if not connected, the user is redirected on onelogin authentification page, if connected, the ressource is accessible) and to have needed informations returned in SAML response (like username and other need data) in the header of the request (to be abble to have them in Spring app).

With this solution, is it possible to keep httpBasic authentification in Spring security and to have "Basic XXXX" in the header of each request set by Shibboleth? Or, have I to remove the httpBasic authentification from Spring Security?

For the second solution (plugin to manage SAML in Spring Security), is it the same result as the first solution and how it must be implemented?

Thank you in advance for your reply.


Solution

  • welcome to stackoverflow.

    ... and to have needed informations returned in SAML response (like username and other need data) in the header of the request (to be abble to have them in Spring app)

    If I understood correctly, you are already using spring security. This means your application is already using spring security populated context for authentication and authorization in your controller/service layers. If you use said approach, where apache is populating the authenticate user information in headers, than this is NOT going to populate the spring security context all by itself UNLESS you add a preAuthFilter in your chain to extract this information and populate your spring context appropriately.

    With this solution, is it possible to keep httpBasic authentification in Spring security and to have "Basic XXXX" in the header of each request set by Shibboleth? Or, have I to remove the httpBasic authentification from Spring Security?

    If you are able to do it then what I said above would be a bit relaxed. Having said that, to best of my knowledge, there is no option where you can deduce a Basic authentication header using shibboleth apache module. In addition, I'll also advice to be careful with this approach since, with this approach, you'll still have to authenticate the user in your app with a dummy password (since you are NOT going to get user's correct password via SAML in this header) and this opens up your application for security exploits. I'll strongly advise against this approach. Shibboleth already has some Spoof Checking covered in their documentation. 

    [EDIT] Based on the additional information, following is what you can do to achieve all handling by apache and still use spring security effectively

    First provide implementation of PreAuthenticatedAuthenticationToken in your application, you can use AbstractPreAuthenticatedProcessingFilter for this purpose. A skeleton for the implementation is provided below, this is excerpt from one of my past work and very much stripped down keeping only the essential elements which are relevant for your scenario. Also take a close look at AuthenticationManager and Authentication docs and make sure you fully understand what to use and for what purpose. Please read javadocs for all these 4 classes carefully to understand the contract as it can be confusing to get it right in spring security otherwise. I have added necessary details as TODO and comments in skeleton blow that you'll have to fill in yourself in your implementation.

    public class ShibbolethAuthFilter extends AbstractPreAuthenticatedProcessingFilter {
        private final String containsValidPrincipalHeader = "_valid_shibboleth_header_present";
        private final String shibbolethHeader = "_shibboleth_header";
        private Logger logger = LoggerFactory.getLogger(getClass());
        
        /**
         * This authentication manager's authenticate method MUST return a fully populated
         * org.springframework.security.core.Authentication object. You may very well use
         * either PreAuthenticatedAuthenticationToken OR UsernamePasswordAuthenticationToken
         * with any credentials set, most important is to correctly populate the Authorities
         * in the returned object so that hasAuthority checks works as expected.
         *
         * Another point, you can use authentication.getPrincipal() in the implementation
         * of authenticate method to access the same principal object as returned by
         * getPreAuthenticatedPrincipal method of this bean. So basically you pass the 
         * using Principal object from this bean to AuthenticationManager's authenticate
         * method which in turn return a fully populated spring's Authentication object
         * with fully populated Authorities.
         */
        @Autowired
        private ShibbolethAuthenticationManager authenticationManager;
    
        @Override
        public void afterPropertiesSet() {
            setAuthenticationManager(authenticationManager);
            super.afterPropertiesSet();
        }
    
        @Override
        protected Object getPreAuthenticatedPrincipal(HttpServletRequest request) {
            String authHeader = request.getHeader(shibbolethHeader);
            if (authHeader == null) {
                logger.trace("No {} header found, skipping Shibboleth Authentication", shibbolethHeader);
                return null;
            }
            // TODO - validate if all header and it's contents are what they should be
            ShibbolethAuthToken authToken = /* TODO - provide your own impl to supply java.security.Principal object here */;
            request.setAttribute(containsValidPrincipalHeader, Boolean.TRUE);
            return authToken;
        }
    
        /**
         * No password required thus Credentials will return null
         */
        @Override
        protected Object getPreAuthenticatedCredentials(HttpServletRequest request) {
            if (Boolean.TRUE.equals(request.getAttribute(containsValidPrincipalHeader)))
                return System.currentTimeMillis(); // just returning non null value to satisfy spring security contract
            logger.trace("Returning null Credentials for non authenticated request");
            return null;
        }
    }
    

    Register this as servlet filter in your app using following registrar

    @Configuration
    public class ShibbolethFilterRegistrar {
    
        /*
         * We don't want to register Shibboleth Filter in spring global chain thus
         * added this explicit registration bean to disable just that.
         */
        @Bean
        public FilterRegistrationBean shibbolethFilterRegistrar(Shibboleth shibbolethAuthFilter) {
            FilterRegistrationBean registration = new FilterRegistrationBean(shibbolethAuthFilter);
            registration.setEnabled(false);
            return registration;
        }
    
        @Bean
        public ShibbolethAuthFilter shibbolethAuthFilter() {
            return new ShibbolethAuthFilter();
        }
    }
    

    Followed by this, change your WebSecurityConfig to following

    @Override
    protected void configure(HttpSecurity http) throws Exception {
        /* autowire shibbolethAuthFilter bean as well */
        http
        .addFilterBefore(shibbolethAuthFilter, AbstractPreAuthenticatedProcessingFilter.class);
        .authorizeRequests()
        .antMatchers("/uri1/**").hasAuthority(Permission.AUTHORITY1.toString())
        .antMatchers("/uri2/**").hasAuthority(Permission.AUTHORITY2.toString())
        .anyRequest().hasAuthority(Permission.AUTHORITY3.toString())
        .and()
        .realmName("App").and().csrf().disable();
        http.authorizeRequests();
        http.headers().frameOptions().sameOrigin().cacheControl().disable();
    }
    

    Hope these pointers helps you to integrate external auth successfully.

    IMHO, following is still valid - as much as I have understood your scenario, if I had to do it, I'll personally prefer to use spring security inbuilt SAML auth for this purpose since that provides very smooth integration with spring security in every possible context within the framework. In addition, it also simplifies my deployment scenario where I'll also have to take care of provisioning apache which'll typically fall under additional workload for DevOps team. For simplicity and scalability, spring security inbuilt SAML SSO support would be my first choice unless there's a constraint which is forcing me to do otherwise (which I am not able to see in current discussion context based on the explanation provided). There are ample tutorials and examples available on net to get it done. I know this is not what you asked for but I thought to share with you what I have done myself in past for similar SSO solutions in spring distributed apps and learning that I had. Hope it helps!!