转载:Shiro自定义过滤器匹配异常

自己追踪了很久的源码,最终还是没找到原因,直到看到这篇文章,赞,学到很多,码了。

原文:http://www.hillfly.com/2017/179.html

 

最近忙着研究在Springboot上使用Shiro的问题。刚好就遇到个诡异事,百度Google也没找到啥有价值的信息,几番周折自己解决了,这里稍微记录下。

自定义过滤器TOC

Shiro支持自定义过滤器大家都知道,也经常用,这里我也用到了一个自定义过滤器,主要用于验证接口调用的AccessToken是否有效。

// AccessTokenFilter.javapublic class AccessTokenFilter extends AccessControlFilter {@Overrideprotected boolean isAccessAllowed(ServletRequest servletRequest,ServletResponse servletResponse,Object o) {if (isValidAccessToken(request)) {return true;}return false;}@Overrideprotected boolean onAccessDenied(ServletRequest servletRequest, ServletResponse servletResponse) throws Exception {throw new UnAuthorizedException("操作授权失败!" + SysConstant.ACCESSTOKEN + "失效!");}
}
// ShiroConfiguration.java@Bean
public AccessTokenFilter accessTokenFilter(){return new AccessTokenFilter();
}@Bean
public ShiroFilterFactoryBean shiroFilter(SecurityManager securityManager,IUrlFilterService urlFilterService) {ShiroFilterFactoryBean shiroFilterFactoryBean = new ShiroFilterFactoryBean();shiroFilterFactoryBean.setSecurityManager(securityManager);// 自定义过滤器Map filterMap = shiroFilterFactoryBean.getFilters();filterMap.put("hasToken", accessTokenFilter());shiroFilterFactoryBean.setFilters(filterMap);// URL过滤Map filterChainDefinitionMap = new LinkedHashMap<>();List urlFilterList = urlFilterService.selectAll();for (UrlFilter filter : urlFilterList) {filterChainDefinitionMap.put(filter.getFilterUrl(),filter.getFilterList());}shiroFilterFactoryBean.setFilterChainDefinitionMap(filterChainDefinitionMap);return shiroFilterFactoryBean;
}

ShiroFilter中的FilterChain是从数据库读取的,如下:

idurlfiltersort
1/druid/**anon1
2/api/loginanon2
3/**hasToken,authc3

我们想要达到的效果是,除了登陆和访问Druid监控页面外,访问其它地址一律要先验证Token,即走我们的自定义过滤器。
修改完毕后启动无异常,我们访问地址验证下。

POST:/api/login

{"hasError": true,"errors": {"httpStatus": 401,"errorCode": "4001","errorMsg": "授权异常:操作授权失败!AccessToken失效!","timestamp": "2017-06-10 11:08:03"}
}

搞笑,结果出乎意料,居然登陆接口走了咱们的那个自定义Filter ??黑人问号脸...

问题排查TOC

FilterChain TOC

首先检查Shiro FilterChain加载的顺序是否异常
1.JPG-45.6kB
.1,集合容器使用LinkedHashMap,保证的FilterChain的顺序
.2,从数据库读取过滤时也是按排序排序的。
从调试结果来看,加载顺序和数据并没有任何问题,都是正确的。

排除了自身的数据问题,那就要往深处挖掘原因了,有了之前解决Quartz问题的经历,这次毫不犹豫就决定跟源码跟踪过滤注册到匹配的过程。

过滤注册TOC

要查明白为何匹配异常,就要先弄清楚咱们的自定义过滤是如何注册到Shiro的,显然,问题的关键在于ShiroFilter返回的ShiroFilterFactoryBean这个类中,我们打开看看。很快,我们就锁定了关键方法:

//ShiroFilterFactoryBean.javaprotected AbstractShiroFilter createInstance() throws Exception {log.debug("Creating Shiro Filter instance.");SecurityManager securityManager = this.getSecurityManager();String msg;if(securityManager == null) {msg = "SecurityManager property must be set.";throw new BeanInitializationException(msg);} else if(!(securityManager instanceof WebSecurityManager)) {msg = "The security manager does not implement the WebSecurityManager interface.";throw new BeanInitializationException(msg);} else {FilterChainManager manager = this.createFilterChainManager();PathMatchingFilterChainResolver chainResolver = new PathMatchingFilterChainResolver();chainResolver.setFilterChainManager(manager);return new ShiroFilterFactoryBean.SpringShiroFilter((WebSecurityManager)securityManager, chainResolver);}
}protected FilterChainManager createFilterChainManager() {DefaultFilterChainManager manager = new DefaultFilterChainManager();Map defaultFilters = manager.getFilters();Iterator var3 = defaultFilters.values().iterator();while(var3.hasNext()) {Filter filter = (Filter)var3.next();this.applyGlobalPropertiesIfNecessary(filter);}Map filters = this.getFilters();String name;Filter filter;if(!CollectionUtils.isEmpty(filters)) {for(Iterator var10 = filters.entrySet().iterator(); var10.hasNext(); manager.addFilter(name, filter, false)) {Entry entry = (Entry)var10.next();name = (String)entry.getKey();filter = (Filter)entry.getValue();this.applyGlobalPropertiesIfNecessary(filter);if(filter instanceof Nameable) {((Nameable)filter).setName(name);}}}Map chains = this.getFilterChainDefinitionMap();if(!CollectionUtils.isEmpty(chains)) {Iterator var12 = chains.entrySet().iterator();while(var12.hasNext()) {Entry entry = (Entry)var12.next();String url = (String)entry.getKey();String chainDefinition = (String)entry.getValue();manager.createChain(url, chainDefinition);}}return manager;
}
//DefaultFilterChainManager.java
public DefaultFilterChainManager() {this.addDefaultFilters(false);
}//DefaultFilter.java
public enum DefaultFilter {anon(AnonymousFilter.class),authc(FormAuthenticationFilter.class),authcBasic(BasicHttpAuthenticationFilter.class),logout(LogoutFilter.class),noSessionCreation(NoSessionCreationFilter.class),perms(PermissionsAuthorizationFilter.class),port(PortFilter.class),rest(HttpMethodPermissionFilter.class),roles(RolesAuthorizationFilter.class),ssl(SslFilter.class),user(UserFilter.class);
}

看到这总算弄清楚Shiro加载Filter的顺序:

  1. 加载 DefaultFilter 中的默认 Filter;
  2. 加载自定义 Filter;
  3. 加载 FFilterChainDefinitionMap;

弄清楚了这个过滤器的加载与注册,那这与我们要解决的问题有何关系呢?首先我们怀疑这里获取的过滤器是异常的,调试打个断点看看。
3.JPG-49.3kB
然而奇怪的是,从调试结果来看,一切加载的Filter都如我们预想的那样,并无异常。

过滤匹配TOC

既然基本排除了Filter加载上出现问题的可能,那么就要来排查过滤匹配的问题了。
重点在于AbstractShiroFilter的doFilterInternal(),这里是匹配的起点。

protected void doFilterInternal(ServletRequest servletRequest, ServletResponse servletResponse, final FilterChain chain) throws ServletException, IOException {Throwable t = null;try {final ServletRequest request = this.prepareServletRequest(servletRequest, servletResponse, chain);final ServletResponse response = this.prepareServletResponse(request, servletResponse, chain);Subject subject = this.createSubject(request, response);subject.execute(new Callable() {public Object call() throws Exception {AbstractShiroFilter.this.updateSessionLastAccessTime(request, response);AbstractShiroFilter.this.executeChain(request, response, chain);return null;}});} catch (ExecutionException var8) {t = var8.getCause();} catch (Throwable var9) {t = var9;}if(t != null) {if(t instanceof ServletException) {throw (ServletException)t;} else if(t instanceof IOException) {throw (IOException)t;} else {String msg = "Filtered request failed.";throw new ServletException(msg, t);}}
}protected void executeChain(ServletRequest request, ServletResponse response, FilterChain origChain) throws IOException, ServletException {FilterChain chain = this.getExecutionChain(request, response, origChain);chain.doFilter(request, response);
}

跟踪到最后,会进入到一个关键方法:

//PathMatchingFilterChainResolver.javapublic FilterChain getChain(ServletRequest request, ServletResponse response, FilterChain originalChain) {FilterChainManager filterChainManager = this.getFilterChainManager();if(!filterChainManager.hasChains()) {return null;} else {String requestURI = this.getPathWithinApplication(request);Iterator var6 = filterChainManager.getChainNames().iterator();String pathPattern;do {if(!var6.hasNext()) {return null;}pathPattern = (String)var6.next();} while(!this.pathMatches(pathPattern, requestURI));if(log.isTraceEnabled()) {log.trace("Matched path pattern [" + pathPattern + "] for requestURI [" + requestURI + "].  Utilizing corresponding filter chain...");}return filterChainManager.proxy(originalChain, pathPattern);}
}

显然,这里就是进行URL匹配的地方。难道是这里匹配出异常了?我们打个断点在这里再访问一下。然而怪异出现了,没有进断点,直接返回了异常信息,根本没有进行匹配! !我们再对自定义过滤器断点调试后发现了过滤器调用链如下:
4.JPG-30.3kB

MMP的,完全没有不是按我们预想的那样进行调用。这TM居然是作为Spring的全局Filter被调用了.Shiro的过滤器优先级居然失效了?我们都知道之前在SpringMVC + Shiro时,都会把Shiro的过滤器配置顺序尽量放前,以达到优先加载的目的。难道这里没有走Shiro的匹配是因为这个吗??难道是因为Springboot先加载了我们自定义的过滤器,然后再加载了ShiroFilter吗,然后这个过滤器优先顺序就出问题了?

我们将断点打到ApplicationFilterChain.java的internalDoFilter()中进行验证下:
5.JPG-74.4kB
!!果然啊!咱们的自定义过滤器居然还在ShiroFilter之前,这就导致请求被我们自定义过滤器先消费掉了..ShiroFilter成了摆设。
那么把咱们的豆放到ShiroFilter后面会如何呢?

@Bean
public ShiroFilterFactoryBean shiroFilter(){}@Bean
public AccessTokenFilter accessTokenFilter(){}

6.JPG-75.8kB
果然顺序变了,那么问题解决了吗?
- 没有,问题依旧,咱们的过滤还是跑了,返回了异常。

看来应该不是这里的顺序问题,我们回过头来继续看ApplicationFilterChain.java的internalDoFilter(),系统会注册的过滤器逐一调用,也就是说无论我们的顺序如何,过滤最终都是会被调用的。

问题解决TOC

眼下我暂时有两种办法去解决这个问题:

  1. 修改AccessTokenFilter,在Filter内部加入路径匹配方法对需要验证令牌的路径进行过滤。
  2. 将咱们的自定义过滤器注册到Shiro,不注册到ApplicationFilterChain。

显然方案一是不可取的,这样修改范围过大,得不偿失了。那我们怎么去实现第二个方法呢?SpringBoot提供了FilterRegistrationBean方便我们对Filter进行管理。

@Bean
public FilterRegistrationBean registration(AccessTokenFilter filter) {FilterRegistrationBean registration = new FilterRegistrationBean(filter);registration.setEnabled(false);return registration;
}

将不需要注册的过滤器注入方法即可。这时候再启动项目进行测试,就可以发现过滤器已经不存在咱们的自定义过滤了。

还有个办法不需要使用到FilterRegistrationBean,因为我们将AccessTokenFilter注册为了Bean交给Spring托管了,所以它会被自动注册到FilterChain中,那我们如果不把它注册为Bean就可以避免这个问题了。

/*** 不需要显示注册Bean了
@Bean
public AccessTokenFilter accessTokenFilter(){}
**/@Bean
public ShiroFilterFactoryBean shiroFilter(SecurityManager securityManager,IUrlFilterService urlFilterService) {//省略filterMap.put("hasToken", new AccessTokenFilter());//省略
}


本文来自互联网用户投稿,文章观点仅代表作者本人,不代表本站立场,不承担相关法律责任。如若转载,请注明出处。 如若内容造成侵权/违法违规/事实不符,请点击【内容举报】进行投诉反馈!

相关文章

立即
投稿

微信公众账号

微信扫一扫加关注

返回
顶部