spring security验证流程

工作需要,又弄起了权限的管理。虽然很早以前都了解过基于容器的权限实现方式,但是一直都觉得那东西太简陋了。后来使用liferay时发现它的权限系统的确做得很优秀,感觉这也可能是它做得最出色的地方吧。但是当时只停留在怎么使用及一些与其衔接的关系之上,并没有对其底层进行了解,新到现在的公司后,发现这一课还是得补上。但是令人惊讶的是,目前可用的选择并不多,甚至很少,最有名的当属spring security了,虽然很有名,但是关于这方面的资料并不是很多,应用示例就更少了。还好有中文的官方文档与http://www.family168.com/bbs/发布的简要教程,因此学起来不至于太困难。然后参照了一篇downpour写的spring security文章,因此勉强熟悉了spring security的应用开发,但是基本只停留在勉强会用的基础之上,而且还花了本人不少时间,在一个项目的运用中,自己更是差点没下得了台,惊出了一身冷汗。当时的感觉spring security就是个垃圾东西,运用很复杂,哪怕是做一个只拦截路径的权限系统,也要经过很多步骤。现在熟悉了它的一些流程后,虽然不知道这样的实现方式是否是最合理的,但是的确也有它的道理。现在利用放假期间,可以静下心来,理解一些以前让自己迷惑的东西了。downpour牛人的那篇文章讲得很好,以至于本人着实花了点时间才把它完全熟悉,当前自己以前对acegi并不熟悉。熟悉了那篇文章后,还是有些地方让自己不太理解,其中之一就是spring security是怎样完成用户角色权限验证的。下面就对这人问题进行简单的介绍:

首先这篇文章是基于downpour那篇文章的,其地址为:
http://www.javaeye.com/topic/319965

最先着手就是配置文件,这也是整个spring security最重要的入口点:

Xml代码
  1. ............   
  2.   
  3.     <beans:bean id="authenticationManager"  
  4.         class="org.springframework.security.providers.ProviderManager">  
  5.         <beans:property name="messageSource" ref="messageSource" />  
  6.     beans:bean>  
  7.        
  8.     <beans:bean id="messageSource"  
  9.         class="org.springframework.context.support.ReloadableResourceBundleMessageSource">  
  10.         <beans:property name="basename"  
  11.             value="classpath:org/springframework/security/messages_zh_CN" />  
  12.     beans:bean>  
  13.   
  14.     <authentication-provider user-service-ref="securityManager">  
  15.         <password-encoder hash="md5" />  
  16.     authentication-provider>  
  17.   
  18.   
  19.     <beans:bean id="accessDecisionManager"  
  20.         class="org.springframework.security.vote.AffirmativeBased">  
  21.           
  22.         <beans:property name="allowIfAllAbstainDecisions"  
  23.             value="false" />  
  24.         <beans:property name="decisionVoters">  
  25.             <beans:list>  
  26.   
  27.                 <beans:bean class="org.springframework.security.vote.RoleVoter" />  
  28.                 <beans:bean class="org.springframework.security.vote.AuthenticatedVoter" />  
  29.             beans:list>  
  30.         beans:property>  
  31.     beans:bean>  
  32.   
  33.     <beans:bean id="resourceSecurityInterceptor"  
  34.         class="org.springframework.security.intercept.web.FilterSecurityInterceptor">  
  35.         <beans:property name="authenticationManager" ref="authenticationManager" />  
  36.         <beans:property name="accessDecisionManager" ref="accessDecisionManager" />  
  37.         <beans:property name="objectDefinitionSource"  
  38.             ref="secureResourceFilterInvocationDefinitionSource" />  
  39.               
  40.         <beans:property name="observeOncePerRequest" value="false" />  
  41.         <custom-filter after="LAST" />  
  42.     beans:bean>  
  43.   
  44.     <beans:bean id="secureResourceFilterInvocationDefinitionSource"  
  45.         class="com.javaeye.sample.security.interceptor.SecureResourceFilterInvocationDefinitionSource" />  
  46. ....  



上面的“accessDecisionManager”就是切入点,首先需要说明的是,在验证用户是否能通过验证时,spring security提供了三种策略,分别对应那个策略类:
UnanimousBased.java 只要有一个Voter不能完全通过权限要求,就禁止访问。
AffirmativeBased.java只要有一个Voter可以通过权限要求,就可以访问。
ConsensusBased.java只要通过的Voter比禁止的Voter数目多就可以访问了。

在此说一点,ConsensusBased这个类有点特别,如果通过的票数与禁止的票数相同怎么办?
这个类有个allowIfEqualGrantedDeniedDecisions属性,默认为true,关键代码:

Java代码
  1. if ((grant == deny) && (grant != 0)) {   
  2.             if (this.allowIfEqualGrantedDeniedDecisions) {   
  3.                 return;   
  4.             } else {   
  5.                 throw new AccessDeniedException(messages.getMessage("AbstractAccessDecisionManager.accessDenied",   
  6.                         "Access is denied"));   
  7.             }   
  8.         }  
if ((grant == deny) && (grant != 0)) {
            if (this.allowIfEqualGrantedDeniedDecisions) {
                return;
            } else {
                throw new AccessDeniedException(messages.getMessage("AbstractAccessDecisionManager.accessDenied",
                        "Access is denied"));
            }
        }



上面的代码表示如果allowIfEqualGrantedDeniedDecisions为true而且通过的票数不为0就授权访问。

在提供好验证策略以后,继续关注其是怎么进行验证的。在上面的XML文件中,accessDescisionManager有个allowIfAllAbstainDecisions属性,这个属性的默认值为false,作用见注释处。现在主要关注的是其decisionVoters属性,中文的理解就是投票,RoleVoter.java 主要作用就是完成角色的投票验证,需要注意的是,它验证的角色,名称必须以ROLE_开头,当然这也可以通过配置文件改变,如:

Xml代码
  1. <bean id="roleVoter" class="org.springframework.security.vote.RoleVoter">  
  2.     <property name="rolePrefix" value="AUTH_"/>  
  3. bean>  



至于RoleVoter是如何完成验证的呆会再说,先回顾一下com.javaeye.sample.security.interceptor.SecureResourceFilterInvocationDefinitionSource:

Java代码
  1. .....   
  2.     public ConfigAttributeDefinition getAttributes(Object filter) throws IllegalArgumentException {   
  3.            
  4.         FilterInvocation filterInvocation = (FilterInvocation) filter;   
  5.         String requestURI = filterInvocation.getRequestUrl();   
  6.         Map urlAuthorities = this.getUrlAuthorities(filterInvocation);   
  7.            
  8.         String grantedAuthorities = null;   
  9.         for(Iterator> iter = urlAuthorities.entrySet().iterator(); iter.hasNext();) {   
  10.             Map.Entry entry = iter.next();   
  11.                
  12.             //url表示从资源表取出的值,在这里代表的是相应的URL   
  13.             String url = entry.getKey();   
  14.                
  15.             //这段代码表示数据库内的需要验证的资源URL与当前请求的URL相匹配时进行验证   
  16.             if(urlMatcher.pathMatchesUrl(url, requestURI)) {   
  17.                 //grantedAuthorities表示每个资源对应的角色,如果有多个角色,则以','隔开   
  18.                 grantedAuthorities = entry.getValue();   
  19.                 break;   
  20.             }   
  21.         }   
  22.            
  23.         if(grantedAuthorities != null) {   
  24.             ConfigAttributeEditor configAttrEditor = new ConfigAttributeEditor();   
  25.             configAttrEditor.setAsText(grantedAuthorities);   
  26.             return (ConfigAttributeDefinition) configAttrEditor.getValue();   
  27.         }   
  28.         return null;   
  29.     }   
  30. ....  
.....
    public ConfigAttributeDefinition getAttributes(Object filter) throws IllegalArgumentException {
        
        FilterInvocation filterInvocation = (FilterInvocation) filter;
        String requestURI = filterInvocation.getRequestUrl();
        Map urlAuthorities = this.getUrlAuthorities(filterInvocation);
        
        String grantedAuthorities = null;
        for(Iterator> iter = urlAuthorities.entrySet().iterator(); iter.hasNext();) {
            Map.Entry entry = iter.next();
            
            //url表示从资源表取出的值,在这里代表的是相应的URL
            String url = entry.getKey();
            
            //这段代码表示数据库内的需要验证的资源URL与当前请求的URL相匹配时进行验证
            if(urlMatcher.pathMatchesUrl(url, requestURI)) {
            	//grantedAuthorities表示每个资源对应的角色,如果有多个角色,则以','隔开
                grantedAuthorities = entry.getValue();
                break;
            }
        }
        
        if(grantedAuthorities != null) {
            ConfigAttributeEditor configAttrEditor = new ConfigAttributeEditor();
            configAttrEditor.setAsText(grantedAuthorities);
            return (ConfigAttributeDefinition) configAttrEditor.getValue();
        }
        return null;
    }
....



虽然不重要,但是还是有必要引用一下:

引用

处于继承树顶端的AbstractSecurityInterceptor有三个实现类:

FilterSecurityInterceptor,负责处理FilterInvocation,实现对URL资源的拦截。
MethodSecurityInterceptor,负责处理MethodInvocation,实现对方法调用的拦截。
AspectJSecurityInterceptor,负责处理JoinPoint,主要也是用于对方法调用的拦截。

为了限制用户访问被保护资源,Spring Security提供了一套元数据,用于定义被保护资源的访问权限,这套元数据主要体现为ConfigAttribute和ConfigAttributeDefinition。每个ConfigAttribute中只包含一个字符串,而一个ConfigAttributeDefinition中可以包含多个ConfigAttribute。对于系统来说,每个被保护资源都将对应一个ConfigAttributeDefinition,这个ConfigAttributeDefinition中包含的多个ConfigAttribute就是访问该资源所需的权限。

实际应用中,ConfigAttributeDefinition会保存在ObjectDefinitionSource中,这是一个主要接口,FilterSecurityInterceptor所需的DefaultFilterInvocationDefinitionSource和MethodSecurityInterceptor所需的MethodDefinitionAttributes都实现了这个接口。ObjectDefinitionSource可以看做是Spring Security中权限配置的源头,框架内部所有的验证组件都是从ObjectDefintionSource中获得数据,来对被保护资源进行权限控制的。

为了从xml中将用户配置的访问权限转换成ObjectDefinitionSource类型的对象,Spring Security专门扩展了Spring中提供的PropertyEditor实现了ConfigAttributeEditor,它可以把以逗号分隔的一系列字符串转换成包含多个ConfigAttribute的ConfigAttributeDefintion对象。

"ROLE_ADMIN,ROLE_USER"



ConfigAttributeDefinition
  ConfigAttribute["ROLE_ADMIN"]
  ConfigAttribute["ROLE_USER"]
       
对于FilterSecurityInterceptor来说,最终生成的就是一个包含了url pattern和ConfigAttributeConfiguration的ObjectDefinitionSource。






                    ConfigAttributeDefinition
"/admin.jsp"   →     ConfigAttribute["ROLE_ADMIN"]
                      ConfigAttribute["ROLE_USER"]
       
换而言之,无论我们将权限配置的原始数据保存在什么地方,只要最终可以将其转换为ObjectDefintionSource就可以提供给验证组件进行调用,实现权限控制。



当时一直不明白这个getAttributes到底拿来做什么的。下面一步步进行追终,通过配置文件可知,这个类首先会到org.springframework.security.intercept.web.FilterSecurityInterceptor这个类中,这个类有个主要的方法:

Java代码
  1. public void invoke(FilterInvocation fi) throws IOException, ServletException {   
  2.        if ((fi.getRequest() != null) && (fi.getRequest().getAttribute(FILTER_APPLIED) != null)   
  3.            && observeOncePerRequest) {   
  4.            // filter already applied to this request and user wants us to observce   
  5.            // once-per-request handling, so don't re-do security checking   
  6.            fi.getChain().doFilter(fi.getRequest(), fi.getResponse());   
  7.        } else {   
  8.            // first time this request being called, so perform security checking   
  9.            if (fi.getRequest() != null) {   
  10.                fi.getRequest().setAttribute(FILTER_APPLIED, Boolean.TRUE);   
  11.            }   
  12.   
  13.            InterceptorStatusToken token = super.beforeInvocation(fi);   
  14.   
  15.            try {   
  16.                fi.getChain().doFilter(fi.getRequest(), fi.getResponse());   
  17.            } finally {   
  18.                super.afterInvocation(token, null);   
  19.            }   
  20.        }   
  21.    }  
 public void invoke(FilterInvocation fi) throws IOException, ServletException {
        if ((fi.getRequest() != null) && (fi.getRequest().getAttribute(FILTER_APPLIED) != null)
            && observeOncePerRequest) {
            // filter already applied to this request and user wants us to observce
            // once-per-request handling, so don't re-do security checking
            fi.getChain().doFilter(fi.getRequest(), fi.getResponse());
        } else {
            // first time this request being called, so perform security checking
            if (fi.getRequest() != null) {
                fi.getRequest().setAttribute(FILTER_APPLIED, Boolean.TRUE);
            }

            InterceptorStatusToken token = super.beforeInvocation(fi);

            try {
                fi.getChain().doFilter(fi.getRequest(), fi.getResponse());
            } finally {
                super.afterInvocation(token, null);
            }
        }
    }



在此有两点需要说明,首先是observeOncePerRequest属性,一般情况下保持其默认的true,文档上有说:

Java代码
  1. /***   
  2. Indicates whether once-per-request handling will be observed. By default this is true,meaning the FilterSecurityInterceptor will only execute once-per-request. Sometimes users may wishit to execute more than once per request, such as when JSP forwards are being used and filter security is desired on each included fragment of the HTTP request.  
  3. */  
/*** 
Indicates whether once-per-request handling will be observed. By default this is true,meaning the FilterSecurityInterceptor will only execute once-per-request. Sometimes users may wishit to execute more than once per request, such as when JSP forwards are being used and filter security is desired on each included fragment of the HTTP request.
*/



其次我们需要关注的重点是FilterSecurityInterceptor的超类AbstractSecurityInterceptor的beforeInvocation方法,下面贴出这个类中我们最需要关注的代码:

Java代码
  1. protected InterceptorStatusToken beforeInvocation(Object object) {   
  2.         Assert.notNull(object, "Object was null");   
  3.   
  4.         if (!getSecureObjectClass().isAssignableFrom(object.getClass())) {   
  5.             throw new IllegalArgumentException("Security invocation attempted for object "  
  6.                     + object.getClass().getName()   
  7.                     + " but AbstractSecurityInterceptor only configured to support secure objects of type: "  
  8.                     + getSecureObjectClass());   
  9.         }   
  10.   
  11.         ConfigAttributeDefinition attr = this.obtainObjectDefinitionSource().getAttributes(object);   
  12.   
  13.         if (attr == null) {   
  14.             if (rejectPublicInvocations) {   
  15.                 throw new IllegalArgumentException(   
  16.                         "No public invocations are allowed via this AbstractSecurityInterceptor. "  
  17.                                 + "This indicates a configuration error because the "  
  18.                                 + "AbstractSecurityInterceptor.rejectPublicInvocations property is set to 'true'");   
  19.             }   
  20.   
  21.             if (logger.isDebugEnabled()) {   
  22.                 logger.debug("Public object - authentication not attempted");   
  23.             }   
  24.   
  25.             publishEvent(new PublicInvocationEvent(object));   
  26.   
  27.             return null// no further work post-invocation   
  28.         }   
  29.   
  30.         if (logger.isDebugEnabled()) {   
  31.             logger.debug("Secure object: " + object + "; ConfigAttributes: " + attr);   
  32.         }   
  33.   
  34.         if (SecurityContextHolder.getContext().getAuthentication() == null) {   
  35.             credentialsNotFound(messages.getMessage("AbstractSecurityInterceptor.authenticationNotFound",   
  36.                     "An Authentication object was not found in the SecurityContext"), object, attr);   
  37.         }   
  38.   
  39.         Authentication authenticated = authenticateIfRequired();   
  40.   
  41.         // Attempt authorization   
  42.         try {   
  43.             this.accessDecisionManager.decide(authenticated, object, attr);   
  44.         }   
  45.         catch (AccessDeniedException accessDeniedException) {   
  46.             AuthorizationFailureEvent event = new AuthorizationFailureEvent(object, attr, authenticated,   
  47.                     accessDeniedException);   
  48.             publishEvent(event);   
  49.   
  50.             throw accessDeniedException;   
  51.         }   
  52.   
  53.         if (logger.isDebugEnabled()) {   
  54.             logger.debug("Authorization successful");   
  55.         }   
  56.   
  57.         AuthorizedEvent event = new AuthorizedEvent(object, attr, authenticated);   
  58.         publishEvent(event);   
  59.   
  60.         // Attempt to run as a different user   
  61.         Authentication runAs = this.runAsManager.buildRunAs(authenticated, object, attr);   
  62.   
  63.         if (runAs == null) {   
  64.             if (logger.isDebugEnabled()) {   
  65.                 logger.debug("RunAsManager did not change Authentication object");   
  66.             }   
  67.   
  68.             // no further work post-invocation   
  69.             return new InterceptorStatusToken(authenticated, false, attr, object);   
  70.         } else {   
  71.             if (logger.isDebugEnabled()) {   
  72.                 logger.debug("Switching to RunAs Authentication: " + runAs);   
  73.             }   
  74.   
  75.             SecurityContextHolder.getContext().setAuthentication(runAs);   
  76.   
  77.             // revert to token.Authenticated post-invocation   
  78.             return new InterceptorStatusToken(authenticated, true, attr, object);   
  79.         }   
  80.     }   
  81.   
  82.     /**  
  83.      * Checks the current authentication token and passes it to the AuthenticationManager if  
  84.      * {@link org.springframework.security.Authentication#isAuthenticated()} returns false or the property  
  85.      * alwaysReauthenticate has been set to true.  
  86.      *  
  87.      * @return an authenticated Authentication object.  
  88.      */  
  89.     private Authentication authenticateIfRequired() {   
  90.         Authentication authentication = SecurityContextHolder.getContext().getAuthentication();   
  91.   
  92.         if (authentication.isAuthenticated() && !alwaysReauthenticate) {   
  93.             if (logger.isDebugEnabled()) {   
  94.                 logger.debug("Previously Authenticated: " + authentication);   
  95.             }   
  96.   
  97.             return authentication;   
  98.         }  
protected InterceptorStatusToken beforeInvocation(Object object) {
        Assert.notNull(object, "Object was null");

        if (!getSecureObjectClass().isAssignableFrom(object.getClass())) {
            throw new IllegalArgumentException("Security invocation attempted for object "
                    + object.getClass().getName()
                    + " but AbstractSecurityInterceptor only configured to support secure objects of type: "
                    + getSecureObjectClass());
        }

        ConfigAttributeDefinition attr = this.obtainObjectDefinitionSource().getAttributes(object);

        if (attr == null) {
            if (rejectPublicInvocations) {
                throw new IllegalArgumentException(
                        "No public invocations are allowed via this AbstractSecurityInterceptor. "
                                + "This indicates a configuration error because the "
                                + "AbstractSecurityInterceptor.rejectPublicInvocations property is set to 'true'");
            }

            if (logger.isDebugEnabled()) {
                logger.debug("Public object - authentication not attempted");
            }

            publishEvent(new PublicInvocationEvent(object));

            return null; // no further work post-invocation
        }

        if (logger.isDebugEnabled()) {
            logger.debug("Secure object: " + object + "; ConfigAttributes: " + attr);
        }

        if (SecurityContextHolder.getContext().getAuthentication() == null) {
            credentialsNotFound(messages.getMessage("AbstractSecurityInterceptor.authenticationNotFound",
                    "An Authentication object was not found in the SecurityContext"), object, attr);
        }

        Authentication authenticated = authenticateIfRequired();

        // Attempt authorization
        try {
            this.accessDecisionManager.decide(authenticated, object, attr);
        }
        catch (AccessDeniedException accessDeniedException) {
            AuthorizationFailureEvent event = new AuthorizationFailureEvent(object, attr, authenticated,
                    accessDeniedException);
            publishEvent(event);

            throw accessDeniedException;
        }

        if (logger.isDebugEnabled()) {
            logger.debug("Authorization successful");
        }

        AuthorizedEvent event = new AuthorizedEvent(object, attr, authenticated);
        publishEvent(event);

        // Attempt to run as a different user
        Authentication runAs = this.runAsManager.buildRunAs(authenticated, object, attr);

        if (runAs == null) {
            if (logger.isDebugEnabled()) {
                logger.debug("RunAsManager did not change Authentication object");
            }

            // no further work post-invocation
            return new InterceptorStatusToken(authenticated, false, attr, object);
        } else {
            if (logger.isDebugEnabled()) {
                logger.debug("Switching to RunAs Authentication: " + runAs);
            }

            SecurityContextHolder.getContext().setAuthentication(runAs);

            // revert to token.Authenticated post-invocation
            return new InterceptorStatusToken(authenticated, true, attr, object);
        }
    }

    /**
     * Checks the current authentication token and passes it to the AuthenticationManager if
     * {@link org.springframework.security.Authentication#isAuthenticated()} returns false or the property
     * alwaysReauthenticate has been set to true.
     *
     * @return an authenticated Authentication object.
     */
    private Authentication authenticateIfRequired() {
        Authentication authentication = SecurityContextHolder.getContext().getAuthentication();

        if (authentication.isAuthenticated() && !alwaysReauthenticate) {
            if (logger.isDebugEnabled()) {
                logger.debug("Previously Authenticated: " + authentication);
            }

            return authentication;
        }



上面的代码首先关注其中的一行代码:

Java代码
  1. ConfigAttributeDefinition attr = this.obtainObjectDefinitionSource().getAttributes(object);  
ConfigAttributeDefinition attr = this.obtainObjectDefinitionSource().getAttributes(object);



不错,这行代码就是SecureResourceFilterInvocationDefinitionSource存在的主要目的,它主要提供URL与ROLE这两个东西,至于需要验证的用户来源,上面有句代码:

Java代码
  1. Authentication authentication = SecurityContextHolder.getContext().getAuthentication();  
Authentication authentication = SecurityContextHolder.getContext().getAuthentication();



众所周知,用户登陆成功后,可以通过:

Java代码
  1. (User) SecurityContextHolder.getContext().getAuthentication().getPrincipal();  
(User) SecurityContextHolder.getContext().getAuthentication().getPrincipal();



上面的代码可以获取当前登陆的用户的基本信息,所以验证时需要它也很自然的。既然所需要的东西都具备了,下面就讲讲该怎么验证的问题了。在上面的AbstractSecurityInterceptor内,有句代码:

Java代码
  1. this.accessDecisionManager.decide(authenticated, object, attr);  
this.accessDecisionManager.decide(authenticated, object, attr);



上面这个accessDecisionManager就是最开始讲的那个accessDecisionManager,终于回到原来的问题上了,由于上面的配置文件中使用了AffirmativeBased投票策略,下面就直接进入此类的decide方法:

Java代码
  1. public void decide(Authentication authentication, Object object, ConfigAttributeDefinition config)   
  2.         throws AccessDeniedException {   
  3.         Iterator iter = this.getDecisionVoters().iterator();   
  4.         int deny = 0;   
  5.   
  6.         while (iter.hasNext()) {   
  7.             AccessDecisionVoter voter = (AccessDecisionVoter) iter.next();   
  8.             int result = voter.vote(authentication, object, config);   
  9.   
  10.             switch (result) {   
  11.             case AccessDecisionVoter.ACCESS_GRANTED:   
  12.                 return;   
  13.   
  14.             case AccessDecisionVoter.ACCESS_DENIED:   
  15.                 deny++;   
  16.   
  17.                 break;   
  18.   
  19.             default:   
  20.                 break;   
  21.             }   
  22.         }   
  23.   
  24.         if (deny > 0) {   
  25.             throw new AccessDeniedException(messages.getMessage("AbstractAccessDecisionManager.accessDenied",   
  26.                     "Access is denied"));   
  27.         }   
  28.   
  29.         // To get this far, every AccessDecisionVoter abstained   
  30.         checkAllowIfAllAbstainDecisions();   
  31.     }  
public void decide(Authentication authentication, Object object, ConfigAttributeDefinition config)
        throws AccessDeniedException {
        Iterator iter = this.getDecisionVoters().iterator();
        int deny = 0;

        while (iter.hasNext()) {
            AccessDecisionVoter voter = (AccessDecisionVoter) iter.next();
            int result = voter.vote(authentication, object, config);

            switch (result) {
            case AccessDecisionVoter.ACCESS_GRANTED:
                return;

            case AccessDecisionVoter.ACCESS_DENIED:
                deny++;

                break;

            default:
                break;
            }
        }

        if (deny > 0) {
            throw new AccessDeniedException(messages.getMessage("AbstractAccessDecisionManager.accessDenied",
                    "Access is denied"));
        }

        // To get this far, every AccessDecisionVoter abstained
        checkAllowIfAllAbstainDecisions();
    }



这个方法主要有三个作用,第一作用就是完成投票,第二就是验证,从上面的switch与if语句可以看出,只要有一个投票者赞成,就直接返回,验证通过。如果没有一个投票者赞成(弃权)而且有人反对,deny++,到if(deny>0)时扔出异常,最后禁止访问。最后一句

Java代码
  1. //到这步时,所有的投票者都弃权了   
  2. // To get this far, every AccessDecisionVoter abstained   
  3.         checkAllowIfAllAbstainDecisions();  
//到这步时,所有的投票者都弃权了
// To get this far, every AccessDecisionVoter abstained
        checkAllowIfAllAbstainDecisions();



这就到了allowIfAllAbstainDecisions属性起作用的时候了。下面就来讲讲AffirmativeBased中用到的投票类RoleVoter,这个类主要工作就是完成投票工作,然后将结果反馈给AffirmativeBased,下面列出RoleVoter的代码:

Java代码
  1. public class RoleVoter implements AccessDecisionVoter {   
  2.     //~ Instance fields ================================================================================================   
  3.   
  4.     private String rolePrefix = "ROLE_";   
  5.   
  6.     //~ Methods ========================================================================================================   
  7.   
  8.     public String getRolePrefix() {   
  9.         return rolePrefix;   
  10.     }   
  11.   
  12.     /**  
  13.      * Allows the default role prefix of ROLE_ to be overridden.  
  14.      * May be set to an empty value, although this is usually not desirable.  
  15.      *  
  16.      * @param rolePrefix the new prefix  
  17.      */  
  18.     public void setRolePrefix(String rolePrefix) {   
  19.         this.rolePrefix = rolePrefix;   
  20.     }   
  21.   
  22.     public boolean supports(ConfigAttribute attribute) {   
  23.         if ((attribute.getAttribute() != null) && attribute.getAttribute().startsWith(getRolePrefix())) {   
  24.             return true;   
  25.         }   
  26.         else {   
  27.             return false;   
  28.         }   
  29.     }   
  30.   
  31.     /**  
  32.      * This implementation supports any type of class, because it does not query  
  33.      * the presented secure object.  
  34.      *  
  35.      * @param clazz the secure object  
  36.      *  
  37.      * @return always true  
  38.      */  
  39.     public boolean supports(Class clazz) {   
  40.         return true;   
  41.     }   
  42.   
  43.     public int vote(Authentication authentication, Object object, ConfigAttributeDefinition config) {   
  44.         int result = ACCESS_ABSTAIN;   
  45.         Iterator iter = config.getConfigAttributes().iterator();   
  46.         GrantedAuthority[] authorities = extractAuthorities(authentication);           
  47.   
  48.         while (iter.hasNext()) {   
  49.             ConfigAttribute attribute = (ConfigAttribute) iter.next();   
  50.   
  51.             if (this.supports(attribute)) {   
  52.                 result = ACCESS_DENIED;   
  53.   
  54.                 // Attempt to find a matching granted authority   
  55.                 for (int i = 0; i < authorities.length; i++) {   
  56.                     if (attribute.getAttribute().equals(authorities[i].getAuthority())) {   
  57.                         return ACCESS_GRANTED;   
  58.                     }   
  59.                 }   
  60.             }   
  61.         }   
  62.   
  63.         return result;   
  64.     }   
  65.        
  66.     GrantedAuthority[] extractAuthorities(Authentication authentication) {   
  67.         return authentication.getAuthorities();   
  68.     }   
  69. }  
public class RoleVoter implements AccessDecisionVoter {
    //~ Instance fields ================================================================================================

    private String rolePrefix = "ROLE_";

    //~ Methods ========================================================================================================

    public String getRolePrefix() {
        return rolePrefix;
    }

    /**
     * Allows the default role prefix of ROLE_ to be overridden.
     * May be set to an empty value, although this is usually not desirable.
     *
     * @param rolePrefix the new prefix
     */
    public void setRolePrefix(String rolePrefix) {
        this.rolePrefix = rolePrefix;
    }

    public boolean supports(ConfigAttribute attribute) {
        if ((attribute.getAttribute() != null) && attribute.getAttribute().startsWith(getRolePrefix())) {
            return true;
        }
        else {
            return false;
        }
    }

    /**
     * This implementation supports any type of class, because it does not query
     * the presented secure object.
     *
     * @param clazz the secure object
     *
     * @return always true
     */
    public boolean supports(Class clazz) {
        return true;
    }

    public int vote(Authentication authentication, Object object, ConfigAttributeDefinition config) {
        int result = ACCESS_ABSTAIN;
        Iterator iter = config.getConfigAttributes().iterator();
        GrantedAuthority[] authorities = extractAuthorities(authentication);        

        while (iter.hasNext()) {
            ConfigAttribute attribute = (ConfigAttribute) iter.next();

            if (this.supports(attribute)) {
                result = ACCESS_DENIED;

                // Attempt to find a matching granted authority
                for (int i = 0; i < authorities.length; i++) {
                    if (attribute.getAttribute().equals(authorities[i].getAuthority())) {
                        return ACCESS_GRANTED;
                    }
                }
            }
        }

        return result;
    }
    
    GrantedAuthority[] extractAuthorities(Authentication authentication) {
    	return authentication.getAuthorities();
    }
}



这个类中最重要的代码就是这句:

Java代码
  1. if (attribute.getAttribute().equals(authorities[i].getAuthority())) {   
  2.                         return ACCESS_GRANTED;   
  3.                     }  
if (attribute.getAttribute().equals(authorities[i].getAuthority())) {
                        return ACCESS_GRANTED;
                    }



这句代码的意思就是把SecureResourceFilterInvocationDefinitionSource传入的角色名称与SecurityContextHolder.getContext().getAuthentication()传入的用户所拥有的角色的角色名称相比较,如果相等则通过验证。

在配置文件中还用到了一个投票类AuthenticatedVoter,这个类与RoleVoter属于同级,RoleVoter用来验证角色,那AutherticatedVoter又是用来干什么的呢?
这个类完整代码:

Java代码
  1. public class AuthenticatedVoter implements AccessDecisionVoter {   
  2.     //~ Static fields/initializers =====================================================================================   
  3.   
  4.     public static final String IS_AUTHENTICATED_FULLY = "IS_AUTHENTICATED_FULLY";   
  5.     public static final String IS_AUTHENTICATED_REMEMBERED = "IS_AUTHENTICATED_REMEMBERED";   
  6.     public static final String IS_AUTHENTICATED_ANONYMOUSLY = "IS_AUTHENTICATED_ANONYMOUSLY";   
  7.     //~ Instance fields ================================================================================================   
  8.   
  9.     private AuthenticationTrustResolver authenticationTrustResolver = new AuthenticationTrustResolverImpl();   
  10.   
  11.     //~ Methods ========================================================================================================   
  12.   
  13.     private boolean isFullyAuthenticated(Authentication authentication) {   
  14.         return (!authenticationTrustResolver.isAnonymous(authentication)   
  15.         && !authenticationTrustResolver.isRememberMe(authentication));   
  16.     }   
  17.   
  18.     public void setAuthenticationTrustResolver(AuthenticationTrustResolver authenticationTrustResolver) {   
  19.         Assert.notNull(authenticationTrustResolver, "AuthenticationTrustResolver cannot be set to null");   
  20.         this.authenticationTrustResolver = authenticationTrustResolver;   
  21.     }   
  22.   
  23.     public boolean supports(ConfigAttribute attribute) {   
  24.         if ((attribute.getAttribute() != null)   
  25.             && (IS_AUTHENTICATED_FULLY.equals(attribute.getAttribute())   
  26.             || IS_AUTHENTICATED_REMEMBERED.equals(attribute.getAttribute())   
  27.             || IS_AUTHENTICATED_ANONYMOUSLY.equals(attribute.getAttribute()))) {   
  28.             return true;   
  29.         } else {   
  30.             return false;   
  31.         }   
  32.     }   
  33.   
  34.     /**  
  35.      * This implementation supports any type of class, because it does not query the presented secure object.  
  36.      *  
  37.      * @param clazz the secure object  
  38.      *  
  39.      * @return always true  
  40.      */  
  41.     public boolean supports(Class clazz) {   
  42.         return true;   
  43.     }   
  44.   
  45.     public int vote(Authentication authentication, Object object, ConfigAttributeDefinition config) {   
  46.         int result = ACCESS_ABSTAIN;   
  47.         Iterator iter = config.getConfigAttributes().iterator();   
  48.   
  49.         while (iter.hasNext()) {   
  50.             ConfigAttribute attribute = (ConfigAttribute) iter.next();   
  51.   
  52.             if (this.supports(attribute)) {   
  53.                 result = ACCESS_DENIED;   
  54.   
  55.                 if (IS_AUTHENTICATED_FULLY.equals(attribute.getAttribute())) {   
  56.                     if (isFullyAuthenticated(authentication)) {   
  57.                         return ACCESS_GRANTED;   
  58.                     }   
  59.                 }   
  60.   
  61.                 if (IS_AUTHENTICATED_REMEMBERED.equals(attribute.getAttribute())) {   
  62.                     if (authenticationTrustResolver.isRememberMe(authentication)   
  63.                         || isFullyAuthenticated(authentication)) {   
  64.                         return ACCESS_GRANTED;   
  65.                     }   
  66.                 }   
  67.   
  68.                 if (IS_AUTHENTICATED_ANONYMOUSLY.equals(attribute.getAttribute())) {   
  69.                     if (authenticationTrustResolver.isAnonymous(authentication) || isFullyAuthenticated(authentication)   
  70.                         || authenticationTrustResolver.isRememberMe(authentication)) {   
  71.                         return ACCESS_GRANTED;   
  72.                     }   
  73.                 }   
  74.             }   
  75.         }   
  76.   
  77.         return result;   
  78.     }   
  79. }  
public class AuthenticatedVoter implements AccessDecisionVoter {
    //~ Static fields/initializers =====================================================================================

    public static final String IS_AUTHENTICATED_FULLY = "IS_AUTHENTICATED_FULLY";
    public static final String IS_AUTHENTICATED_REMEMBERED = "IS_AUTHENTICATED_REMEMBERED";
    public static final String IS_AUTHENTICATED_ANONYMOUSLY = "IS_AUTHENTICATED_ANONYMOUSLY";
    //~ Instance fields ================================================================================================

    private AuthenticationTrustResolver authenticationTrustResolver = new AuthenticationTrustResolverImpl();

    //~ Methods ========================================================================================================

    private boolean isFullyAuthenticated(Authentication authentication) {
        return (!authenticationTrustResolver.isAnonymous(authentication)
        && !authenticationTrustResolver.isRememberMe(authentication));
    }

    public void setAuthenticationTrustResolver(AuthenticationTrustResolver authenticationTrustResolver) {
        Assert.notNull(authenticationTrustResolver, "AuthenticationTrustResolver cannot be set to null");
        this.authenticationTrustResolver = authenticationTrustResolver;
    }

    public boolean supports(ConfigAttribute attribute) {
        if ((attribute.getAttribute() != null)
            && (IS_AUTHENTICATED_FULLY.equals(attribute.getAttribute())
            || IS_AUTHENTICATED_REMEMBERED.equals(attribute.getAttribute())
            || IS_AUTHENTICATED_ANONYMOUSLY.equals(attribute.getAttribute()))) {
            return true;
        } else {
            return false;
        }
    }

    /**
     * This implementation supports any type of class, because it does not query the presented secure object.
     *
     * @param clazz the secure object
     *
     * @return always true
     */
    public boolean supports(Class clazz) {
        return true;
    }

    public int vote(Authentication authentication, Object object, ConfigAttributeDefinition config) {
        int result = ACCESS_ABSTAIN;
        Iterator iter = config.getConfigAttributes().iterator();

        while (iter.hasNext()) {
            ConfigAttribute attribute = (ConfigAttribute) iter.next();

            if (this.supports(attribute)) {
                result = ACCESS_DENIED;

                if (IS_AUTHENTICATED_FULLY.equals(attribute.getAttribute())) {
                    if (isFullyAuthenticated(authentication)) {
                        return ACCESS_GRANTED;
                    }
                }

                if (IS_AUTHENTICATED_REMEMBERED.equals(attribute.getAttribute())) {
                    if (authenticationTrustResolver.isRememberMe(authentication)
                        || isFullyAuthenticated(authentication)) {
                        return ACCESS_GRANTED;
                    }
                }

                if (IS_AUTHENTICATED_ANONYMOUSLY.equals(attribute.getAttribute())) {
                    if (authenticationTrustResolver.isAnonymous(authentication) || isFullyAuthenticated(authentication)
                        || authenticationTrustResolver.isRememberMe(authentication)) {
                        return ACCESS_GRANTED;
                    }
                }
            }
        }

        return result;
    }
}



作用见下面引用:

引用

AuthenticatedVoter用于判断ConfigAttribute上是否拥有IS_AUTHENTICATED_FULLY,IS_AUTHENTICATED_REMEMBERED或IS_AUTHENTICATED_ANONYMOUSLY之类的配置。
如果配置为IS_AUTHENTICATED_FULLY,那么只有AuthenticationTrustResolver的isAnonymous()和isRememberMe()都返回false时才能通过验证。
如果配置为IS_AUTHENTICATED_REMEMBERED,那么会在AuthenticationTrustResolver的isAnonymous()返回false时通过验证。
如果配置为IS_AUTHENTICATED_ANONYMOUSLY,就可以在AuthenticationTrustResolver的isAnonymous()和isRememberMe()两个方法返回任意值时都可以通过验证。



其中上面引用中的ConfigAttribute就是指SecureResourceFilterInvocationDefinitionSource的getAttributes()方法中ConfigAttributeDefinition中的ConfigAttribute,downpour的文章只是传入了ROLE_USER,ROLE_ADMIN等内容,要想让AuthenticatedVoter有用武之地,可以传入
IS_AUTHENTICATED_FULLY、IS_AUTHENTICATED_REMEMBERED、IS_AUTHENTICATED_ANONYMOUSLY中的值。

需要注意的是AffirmativeBased中遍历的投票者是要分先后的,也就是说RoleVoter在AuthenticatedVoter前面的话,会先进行RoleVoter验证,如果RoleVoter投票未通过,再进行了AuthenticatedVoter投票。

这样spring security的验证流程就基本清楚了,当然这篇文章也还是有些地方讲得不完善,以事有时间再来修改。

在前一篇文章中,还是有些地方没讲清楚,但那篇文章已经有点长了,所以还是另外单独讲一下吧。见SecureResourceFilterInvocationDefinitionSource代码:

Java代码
  1. /**  
  2.  * RegexUrlPathMatcher默认不进行小写转换,而AntUrlPathMatcher默认要进行小写转换  
  3.  */  
  4.     public void afterPropertiesSet() throws Exception {   
  5.            
  6.         // default url matcher will be RegexUrlPathMatcher   
  7.         this.urlMatcher = new RegexUrlPathMatcher();   
  8.            
  9.         if (useAntPath) {  // change the implementation if required   
  10.             this.urlMatcher = new AntUrlPathMatcher();   
  11.         }   
  12.            
  13.         // Only change from the defaults if the attribute has been set   
  14.         if ("true".equals(lowercaseComparisons)) {   
  15.             if (!this.useAntPath) {   
  16.                 ((RegexUrlPathMatcher) this.urlMatcher).setRequiresLowerCaseUrl(true);   
  17.             }   
  18.         } else if ("false".equals(lowercaseComparisons)) {   
  19.             if (this.useAntPath) {   
  20.                 //是否对URL全部转换成小写格式   
  21.                 ((AntUrlPathMatcher) this.urlMatcher).setRequiresLowerCaseUrl(false);   
  22.             }   
  23.         }   
  24.            
  25.     }   
  26.   
  27. //这个方法主要会在FilterSecurityInterceptor->AbstractSecurityInterceptor->beforeInvocation中用到   
  28.     public ConfigAttributeDefinition getAttributes(Object filter) throws IllegalArgumentException {   
  29.            
  30.         FilterInvocation filterInvocation = (FilterInvocation) filter;   
  31.         String requestURI = filterInvocation.getRequestUrl();   
  32.         Map urlAuthorities = this.getUrlAuthorities(filterInvocation);   
  33.            
  34.         String grantedAuthorities = null;   
  35.         for(Iterator> iter = urlAuthorities.entrySet().iterator(); iter.hasNext();) {   
  36.             Map.Entry entry = iter.next();   
  37.                
  38.             //url表示从资源表取出的值,在这里代表的是相应的URL   
  39.             String url = entry.getKey();   
  40.                
  41.             //这段代码表示数据库内的需要验证的资源URL与当前请求的URL相匹配时进行验证   
  42.             if(urlMatcher.pathMatchesUrl(url, requestURI)) {   
  43.                 //grantedAuthorities表示每个资源对应的角色,如果有多个角色,则以','隔开   
  44.                 grantedAuthorities = entry.getValue();   
  45.                 break;   
  46.             }   
  47.         }   
  48.            
  49.         if(grantedAuthorities != null) {   
  50.             ConfigAttributeEditor configAttrEditor = new ConfigAttributeEditor();   
  51.             configAttrEditor.setAsText(grantedAuthorities);   
  52.             return (ConfigAttributeDefinition) configAttrEditor.getValue();   
  53.         }   
  54.            
  55.         //返回null表示不会验证   
  56.         return null;   
  57.     }  
/**
 * RegexUrlPathMatcher默认不进行小写转换,而AntUrlPathMatcher默认要进行小写转换
 */
    public void afterPropertiesSet() throws Exception {
        
        // default url matcher will be RegexUrlPathMatcher
        this.urlMatcher = new RegexUrlPathMatcher();
        
        if (useAntPath) {  // change the implementation if required
            this.urlMatcher = new AntUrlPathMatcher();
        }
        
        // Only change from the defaults if the attribute has been set
        if ("true".equals(lowercaseComparisons)) {
            if (!this.useAntPath) {
                ((RegexUrlPathMatcher) this.urlMatcher).setRequiresLowerCaseUrl(true);
            }
        } else if ("false".equals(lowercaseComparisons)) {
            if (this.useAntPath) {
            	//是否对URL全部转换成小写格式
                ((AntUrlPathMatcher) this.urlMatcher).setRequiresLowerCaseUrl(false);
            }
        }
        
    }

//这个方法主要会在FilterSecurityInterceptor->AbstractSecurityInterceptor->beforeInvocation中用到
    public ConfigAttributeDefinition getAttributes(Object filter) throws IllegalArgumentException {
        
        FilterInvocation filterInvocation = (FilterInvocation) filter;
        String requestURI = filterInvocation.getRequestUrl();
        Map urlAuthorities = this.getUrlAuthorities(filterInvocation);
        
        String grantedAuthorities = null;
        for(Iterator> iter = urlAuthorities.entrySet().iterator(); iter.hasNext();) {
            Map.Entry entry = iter.next();
            
            //url表示从资源表取出的值,在这里代表的是相应的URL
            String url = entry.getKey();
            
            //这段代码表示数据库内的需要验证的资源URL与当前请求的URL相匹配时进行验证
            if(urlMatcher.pathMatchesUrl(url, requestURI)) {
            	//grantedAuthorities表示每个资源对应的角色,如果有多个角色,则以','隔开
                grantedAuthorities = entry.getValue();
                break;
            }
        }
        
        if(grantedAuthorities != null) {
            ConfigAttributeEditor configAttrEditor = new ConfigAttributeEditor();
            configAttrEditor.setAsText(grantedAuthorities);
            return (ConfigAttributeDefinition) configAttrEditor.getValue();
        }
        
        //返回null表示不会验证
        return null;
    }



这个方法的主要作用是从数据库的resource表加载出所有的资源URL值,通过它与request请求的URL相比较,而比较器一般采用AntUrlPathMatcher,如果需要更加灵活的方式,可以使用RegexUrlPathMatcher,先介绍一下上面的getAttributes是怎么通过URL进行验证的。从上面的for循环可以看出,首先遍历资源URL的列表,如果发现有与当前request请求URL相匹配的,就进行验证,如果当前用户拥有的角色与此资源URL对应的角色相同,则通过验证,否则禁止访问。见下图:



上面一张简单的图,基本能说明意思,需要说明的是只要发现有与当前请求路径相匹配的,就会进行验证,而且只验证一次,没有通过验证也不会再检查是否匹配第二个资源URL,的当然如果请求的URL与第一个资源URL不匹配,就会继续向下查找,直到找到与请求URL匹配的资源URL为止,如果遍历完以后,还是没有查到,就到弃权处理,至于所有投票者都弃权以后,该怎么处理,前一篇博客有介绍的。由此可知,这个资源的URL路径的顺序就比较重要了。如果遍历出来的第一个资源URL为/**,而普通用户角色对应的资源URL没有它,那么即使普通角色拥有其它的资源URL权限也是不能访问到相应的页面的。但是如果普通用户有/**的权限,而/**是匹配所有路径的,如/admin/index.jsp也是会匹配的,这样就相当于普通用户拥有任何路径的权限,这也是不行的。所以遍历出来的资源URL顺序很重要。

在基于XML的方式授权时就是把/**放在最后面的,如:

Xml代码
  1. <intercept-url pattern="/secure/extreme/**" access="ROLE_SUPERVISOR"/>  
  2.         <intercept-url pattern="/secure/**" access="IS_AUTHENTICATED_REMEMBERED" />  
  3.         <intercept-url pattern="/**" access="IS_AUTHENTICATED_ANONYMOUSLY" />  



也就是说,这个路径放的顺序很重要,在数据库里存放URL时也要遵循这样的规则,如上面的XML放在数据库里应该这样:



这里需要多做的一个步骤是,取出来后不要改变取数据库记录的顺序。在SecurityManagerSupport中的代码:

Java代码
  1. public Map loadUrlAuthorities() {   
  2.         Map urlAuthorities = new LinkedHashMap();   
  3.            
  4.         @SuppressWarnings("unchecked")   
  5.         List urlResources = getHibernateTemplate().find("FROM Resource resource WHERE resource.type = ?""URL");   
  6.         for(Resource resource : urlResources) {   
  7.             urlAuthorities.put(resource.getValue(), resource.getRoleAuthorities());   
  8.         }   
  9.         return urlAuthorities;   
  10.     }  
public Map loadUrlAuthorities() {
        Map urlAuthorities = new LinkedHashMap();
        
        @SuppressWarnings("unchecked")
        List urlResources = getHibernateTemplate().find("FROM Resource resource WHERE resource.type = ?", "URL");
        for(Resource resource : urlResources) {
            urlAuthorities.put(resource.getValue(), resource.getRoleAuthorities());
        }
        return urlAuthorities;
    }



要保持取出的数据的顺序不改变,需要使用LinkedHashMap,这样SecureResourceFilterInvocationDefinitionSource在验证URL的顺序就与数据库里面存的顺序一致了,表面上看的确有些不灵活,而实际上,在使用中时资源URL是固定的,用户不能改变的,所以这也没什么影响。

关于AntUrlPathMatcher的匹配规则也很简单,文档中有说明:

Java代码
  1. ? matches one character
  2.   
  3. * matches zero or more characters
  4.   
  5. ** matches zero or more 'directories' in a path
  6.   
 * 
  • ? matches one character
  • *
  • * matches zero or more characters
  • *
  • ** matches zero or more 'directories' in a path
  • 你可能感兴趣的:(JAVA,security,spring,authentication,object,access,url)