Spring Security资源通配符冲突问题

原创 2017年01月03日 13:47:43

1 系统配置

使用数据库配置系统 资源-角色-用户

1 资源

资源a: /admin/*
资源b:/admin/admin!index.action

2 角色

超级管理员:role_super
普通管理员:role_normal

3 资源-角色

role_super拥有资源a
role_normal拥有资源b

2 期望效果

超级管理员能够访问所有以/admin/开头的资源(当然包括资源b),普通管理员只能访问首页:/admin/admin!index.action

3 问题描述

有的时候超级管理员不能访问资源b,但是能访问其它的/admin/开头的资源,普通管理员只能访问资源b。
而有的时候,普通管理员无法访问所有/admin/开头的资源(不能访问资源b),超级管理员能访问所有/admin/开头的资源。

4 源码跟踪

角色是否拥有对某个特定资源的访问权限,取决于以下两个投票器:

    <bean id="accessDecisionManager" class="org.springframework.security.vote.AffirmativeBased">
        <property name="decisionVoters">
            <list>
                <bean class="org.springframework.security.vote.RoleVoter" />
                <bean class="org.springframework.security.vote.AuthenticatedVoter" />
            </list>
        </property>
    </bean>

角色投票器:

org.springframework.security.vote.RoleVoter

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;
}

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

简单的观察代码运行此处的三个变量值:
authentication:当前用户的认证信息
org.springframework.security.providers.UsernamePasswordAuthenticationToken@da7bc0ad: Principal: cc.jiuyi.entity.Admin@617667eb; Password: [PROTECTED]; Authenticated: true; Details: org.springframework.security.ui.WebAuthenticationDetails@380f4: RemoteIpAddress: 127.0.0.1; SessionId: 79F021F693F49C844524B5EECC9B2D08; Granted Authorities: ROLE_NORMAL
object:想要访问的资源
FilterInvocation: URL: /admin/admin!index.action
config:拥有该资源的角色组
[ROLE_ADMIN]
结合代码逻辑,角色投票器的主要作用:
如果没有任何角色拥有该资源,或者所有角色都不是有效角色名,则投弃权票;
否则将用户的角色和角色组一一比对,如果都不匹配,投反对票,若存在匹配,则投赞成票。

5 原因分析

在期望效果中,我希望的逻辑是:
role_super 拥有 admin/*,以及/admin/admin!index.action权限
role_normal 拥有 /admin/admin!index.action权限
但是在spring security的代码中判断逻辑是:
/admin/admin!index.action资源的所属角色有[ROLE_ADMIN]
/admin/admin!index.action资源不是明确赋予了[ROLE_NORMAL]吗?
到这一步,我就没有再细看源码了。猜测的原因是,/admin/admin!index.action匹配了通配符admin/*,spring security就将/admin/admin!index.action这个资源划分到[ROLE_ADMIN],并停止了为它寻找其它的所属。
当然,在下一次启动的时候,/admin/admin!index.action资源的所属角色可能变成了[ROLE_NORMAL]

版权声明:本文为博主原创文章,转载需注明出处。

Spring Security(17)——基于方法的权限控制

之前介绍的都是基于URL的权限控制,Spring Security同样支持对于方法的权限控制。可以通过intercept-methods对某个bean下面的方法进行权限控制,也可以通过pointcut...

spring security3.x学习(21)_关于方法过滤

@RolesAllowed("ROLE_USER") public void changePassword(String username, String password);  或 @Ro...

springSecurity理解一 一个简单的HelloWorld

第 1 章 一个简单的HelloWorld Spring Security中可以使用Acegi-1.x时代的普通配置方式,也可以使用从2.0时代才出现的命名空间配置方式,实际上这两者实现的功能是完全...

《Spring Security3》第二章翻译

第二章  Spring Security起步 在本章中,我们将要学习SpringSecurity背后的核心理念,包括重要的术语和产品架构。我们将会关注配置Spring Security的一些方式以及...

关于Spring Security 3获取用户信息的问题

原文:http://blog.csdn.net/jjk_02027/article/details/6544889 Spring Security 3取不到用户信息的问题 不是惹的祸| ...

Spring antmatcher 使用路径通配符加载Resource,

Spring提供了强大的Ant模式通配符匹配,从同一个路径能匹配一批资源。 Ant路径通配符支持"?"、"*"、"**",注意通配符匹配不包括目录分隔符“/”。 “?”:匹配一个字符,如"conf...

Spring和shiro都用到的Ant 匹配原则!

Ant path 匹配原则 在Spring MVC中经常要用到拦截器,在配置需要要拦截的路径时经常用到子标签,其有一个path属性,它就是用来指定需要拦截的路径的。例如: ...

Spring antmatcher 使用路径通配符加载Resource,

Spring提供了强大的Ant模式通配符匹配,从同一个路径能匹配一批资源。 Ant路径通配符支持"?"、"*"、"**",注意通配符匹配不包括目录分隔符“/”。 “?”:匹配一个字符,如"conf...

Spring Security教程(11)---- 使用数据库来管理资源

这个可以说是SpringSecurity最核心的东西,在项目中资源很多肯定不能一一配置到配置文件中,所以用数据库来管理资源是必然的。这个也很容易实现。表结构已经在之前都创建过了。 首先我们要来从数据...

spring security使用数据库获取资源、角色和权限保护web应用

在《使用数据库定义资源、角色和权限》中已经定义了该示例的实体关系,本文对其进行实现。web应用中有三种资源。 /main/common.action:具有common角色的用户就可以访问 /ma...
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:Spring Security资源通配符冲突问题
举报原因:
原因补充:

(最多只允许输入30个字)