1. 由于最近自己写的一个项目上用到了多realm的使用,遇到了一个这样的问题:

  1. 自己继承了BasicHttpAuthenticationFilter,实现了获取token,然后直接请求api的方法,但是每次第一次调用的时候都是无效的,第二次请求又是正常的。

以下为配置文件

  1. @Bean
  2. public ShiroFilterFactoryBean shirFilter(org.apache.shiro.mgt.SecurityManager securityManager) {
  3. logger.debug("ShiroConfiguration.shiroFilter()");
  4. ShiroFilterFactoryBean shiroFilterFactoryBean = new ShiroFilterFactoryBean();
  5. shiroFilterFactoryBean.setSecurityManager(securityManager);
  6. //拦截器.
  7. Map<String, String> filterChainDefinitionMap = new LinkedHashMap<String, String>();
  8. // 配置不会被拦截的链接 顺序判断
  9. filterChainDefinitionMap.put("/css/**", "anon");
  10. filterChainDefinitionMap.put("/fonts/**", "anon");
  11. filterChainDefinitionMap.put("/img/**", "anon");
  12. filterChainDefinitionMap.put("/js/**", "anon");
  13. filterChainDefinitionMap.put("/index.html", "anon");
  14. filterChainDefinitionMap.put("/login.html", "anon");
  15. filterChainDefinitionMap.put("/register.html", "anon");
  16. //配置退出 过滤器,其中的具体的退出代码Shiro已经替我们实现了
  17. filterChainDefinitionMap.put("/auth/logout", "logout");
  18. filterChainDefinitionMap.put("/auth/login", "anon");
  19. filterChainDefinitionMap.put("/wx/app/login/**", "anon");
  20. filterChainDefinitionMap.put("/auth/register", "anon");
  21. //<!-- 过滤链定义,从上向下顺序执行,一般将/**放在最为下边 -->:这是一个坑呢,一不小心代码就不好使了;
  22. //<!-- authc:所有url都必须认证通过才可以访问; anon:所有url都都可以匿名访问-->
  23. filterChainDefinitionMap.put("/**", "authc,token");
  24. // 如果不设置默认会自动寻找Web工程根目录下的"/login.jsp"页面
  25. //配置shiro默认登录界面地址,前后端分离中登录界面跳转应由前端路由控制,后台仅返回json数据
  26. shiroFilterFactoryBean.setLoginUrl("/unauth");
  27. // 登录成功后要跳转的链接
  28. shiroFilterFactoryBean.setSuccessUrl("/index");
  29. //未授权界面;
  30. shiroFilterFactoryBean.setUnauthorizedUrl("/403");
  31. shiroFilterFactoryBean.setFilterChainDefinitionMap(filterChainDefinitionMap);
  32. Map<String,Filter> filterMap=new HashedMap();
  33. filterMap.put("token",headerHttpAuthenticationFilter());
  34. shiroFilterFactoryBean.setFilters(filterMap);
  35. return shiroFilterFactoryBean;
  36. }

  2. 贴出主要的配置文件

2.分析问题:

  1. 由于第一次不正常,第二次正常,又因为shiro的权限认证是根据sessionId+过滤器实现的,每次删除sessionId的cookie后,第一次通过token方式进行请求都会出现没有权限的问题。

  2. 检查HeaderHttpAuthenticationFilter类发现正常,在该类中打断点,发现方法能够正常进入到该方法中,并且是正常的

3. 查询shiro中的Filter调用链,发现ProxiedFilterChain类中是这样进行调用的

   

  1. public void doFilter(ServletRequest request, ServletResponse response) throws IOException, ServletException {
  2. if (this.filters != null && this.filters.size() != this.index) {
  3. if (log.isTraceEnabled()) {
  4. log.trace("Invoking wrapped filter at index [" + this.index + "]");
  5. }
  6.  
  7. ((Filter)this.filters.get(this.index++)).doFilter(request, response, this);
  8. } else {
  9. if (log.isTraceEnabled()) {
  10. log.trace("Invoking original filter chain.");
  11. }
  12.  
  13. this.orig.doFilter(request, response);
  14. }
  15.  
  16. }

  获取filters然后进行一个一个调用,查询以及跟踪断点发现,authc该方式中对应的为BasicHttpAuthenticationFilter

在BasicHttpAuthenticationFilter打上断点

发现每次会请求到该方法上,查找原因发现,由于我是token请求,并没有带该参数导致的

问题解决:

  1. 查看ShiroConfig类中发现

  1. filterChainDefinitionMap.put("/**", "authc,token");

  位置放置权限filter鉴权有问题,由于token的鉴权会比authc的权限高,自定义权限比普通的表单认证的优先级高。所以应该将其放到前面,修改为如下

 

  1. filterChainDefinitionMap.put("/**", "token,authc");

  2. 重启项目,运行正常了

shiro多Realm第一次调用不生效问题的更多相关文章

  1. Shiro笔记(四)Shiro的realm认证

    认证流程: 1.获取当前Subject.调用SecurityUtils.getSubject(); 2.测试当前用户是否已经被认证,即是否已经登录,调用Subject的isAurhenticated( ...

  2. Shiro中Realm

    6.1 Realm [2.5 Realm]及[3.5 Authorizer]部分都已经详细介绍过Realm了,接下来再来看一下一般真实环境下的Realm如何实现. 1.定义实体及关系   即用户-角色 ...

  3. Shiro自定义Realm时用注解的方式注入父类的credentialsMatcher

    用Shiro做登录权限控制时,密码加密是自定义的. 数据库的密码通过散列获取,如下,算法为:md5,盐为一个随机数字,散列迭代次数为3次,最终将salt与散列后的密码保存到数据库内,第二次登录时将登录 ...

  4. 【转】WCF 服务第一次调用慢的问题

    写了一个WCF Serivces供外部程序通过.NET Businesss Connector调用AX的代码,第一次调用的时候总是很慢,有时候甚至超过1分钟,访问地址改成http://localhos ...

  5. winform客户端程序第一次调用webservice方法很慢的解决方法

    .net2.0的winform客户端最常用的与服务端通信方式是通过webservice,最近在用dottrace对客户端做性能测试的时候发现,客户端程序启动以后,第一次调用某一个webservice的 ...

  6. shiro自定义Realm

    1.1 自定义Realm 上边的程序使用的是shiro自带的IniRealm,IniRealm从ini配置文件中读取用户的信息,大部分情况下需要从系统的数据库中读取用户信息,所以需要自定义realm. ...

  7. SpringCloud服务消费者第一次调用出现超时问题的解决方案

    在第一次访问服务消费者的时候(消费者去调用服务提供者服务)会出现如下异常: com.netflix.hystrix.exception.HystrixRuntimeException: TestSer ...

  8. 百度编辑器同一id重复调用不能生效的办法

    在使用js 调用表单组件模板的时候,表单内有一个编辑框 第一次调出的时候,百度编辑器正常显示,关闭后,再打开,百度编辑器不能显示 原因:第一次使用的时候, UE.getEditor('node_con ...

  9. shiro开启realm

    使用缓存,可以解决每次访问请求都查数据库的问题.第一次授权后存入缓存. 缓存流程 shiro中提供了对认证信息和授权信息的缓存.shiro默认是关闭认证信息缓存的,对于授权信息的缓存shiro默认开启 ...

随机推荐

  1. 阿里、百度等多家公司Java面试记录与总结

    算算自己大概面试了近十家公司,也拿到了几个Offer,现在面试告一段落,简单总结下面试经验. 我现在主要的方向是Java服务端开发,把遇到的问题和大家分享一下,也谈谈关于技术人员如何有方向的提高自己, ...

  2. Ubuntu16.04安装RealSense SR300驱动

    原文链接 https://blog.csdn.net/u013401766/article/details/78472285 第一步:CMake 3.14.0 安装 1)下载cmake-3.14.1. ...

  3. Django学习之十一:真正理解Django的路由分发和反解url原理

    目录 URL Dispatcher 简介 模式概念 对比URLPattern 与 URLResolver (多态的体现) 构建子路由几种方式 反解url算法逻辑 URL Dispatcher 简介 d ...

  4. Deepin Linux系统的日常使用总结(日常施工)

    1.登录root权限用户 sudo su 2.安装软件语句 apt-get install <package_name> 相对的, 安装:apt-get install <packa ...

  5. vue v-for动画bug

    因为是v-for 循环 出来的,:key = "index" 会出现问题,所以,需要把:key="XXX"换成其他属性就好了. 链接参考: https://se ...

  6. Dynamics 365 Customer Engagement中插件的调试

    微软动态CRM专家罗勇 ,回复319或者20190319可方便获取本文,同时可以在第一间得到我发布的最新博文信息,follow me!我的网站是 www.luoyong.me . 本文主要根据官方的教 ...

  7. 【设计模式】原型模式 Pototype Pattern

    前面讲了创建一个对象实例的方法单例模式Singleton Pattern, 创造多个产品的工厂模式(简单工厂模式 Simple Factory Pattern, 工厂方法模式 FactoryMothe ...

  8. typescript中的接口

    说到接口:在面向对象的编程中,接口是一种规范的定义,它定义了行为和动作的规范,在程序设计里面,接口起到一种限制和规范的作用.接口定义了某一批类所需要遵守的规范,接口不关心这些类的内部状态数据,也不关心 ...

  9. Bootstrap table 行编辑导航

    /*开启表格编辑方向键导航 方向键(←): VK_LEFT (37) 方向键(↑): VK_UP (38) 方向键(→): VK_RIGHT (39) 方向键(↓): VK_DOWN (40) */ ...

  10. 在html代码中js的script标签建议放在那里?

    今天编写了一个简单的js代码,F12有错误,然后发现是<script>放的位置有问题.之前在我的印象当中,说的是这个标签放在哪里都可以,然而...并不是这样的,例如我现在练习的这个代码,写 ...