Shiro 学习应用(续)
在前面的文章中为大家介绍了 Shrio 的基础概念。可能比較笼统。没有深入到开发过程的一些问题。如今集中在本帖中归纳一下有关问题。
FormAuthenticationFilter 表单过滤器
表单过滤器的问题,是本人在实现验证码组件时候遇到的,亦以前一度让我“抓狂”。虽然有便捷的方法实现验证码,比如在控制器中就能够推断验证码逻辑了,可是那有违 Shiro 结构体系的思想;本人也想籍此了解 Shiro 扩展实现方法,——假设都走捷径。那么学习的目的就达不到了。于是,本人就把遇到问题逐一罗列出来。
首先啰嗦下。为什么不写在控制器?
Shiro 提供丰富的过滤器,FormAuthenticationFilter 就是一种过滤器。既然使用了安全框架。那么我们就应该依照框架的方法去编码。我们知道,经典 Web MVC 中,过滤器是优于 Servlet 运行的,也是一般用作安全检測、权限校验之用的。
所以 Shiro 组件基于过滤器的思想就“名正言顺”了(所以说 Filter 能做的,Shiro 也能做,Filter 做不了的或者不适合做的, Shiro 过滤器也不会僭越)。实际从作用功效上,过滤器与 Servlet 彼此替代可能性蛮大的。也就是说,Servlet 的逻辑用在 Filter 之上亦能够,效果看起来不会差太多——反之亦然。只是,我们还是要回想下它们之间的差别:首先是前面说的。Filter 优于 Servlet 运行,Filter 拦截了,Servlet 也不会运行;其次,编码风格上 Filter 基于责任链式模式,Servlet 却不是;最后就是 Filter 不像 Servlet,它不能产生一个请求或者响应。Filter 通常仅仅是返回 true/false。当然,它们是如此地相像。以至于把 Filter 觉得是 Servlet 变种的一种亦未尝不可。
所以,我们会看到 Shiro 的一些案例中,既有 Filter。又有 Controller。比如以下我们绑定会员登录的 url 到 FormAuthenticationFilter,然后又声明有控制器。
CaptchaFormAuthenticationFilter 是扩展 FormAuthenticationFilter 的验证码过滤器。
/service/user/access/login = authcaptcha 这句就是绑定了验证码过滤器。
这样如此。应该是不用编码就能够调用 CaptchaFormAuthenticationFilter 了。
这个完整的请求还有 Controller 部分:
@RequestMapping(value = "/login", method = RequestMethod.POST)
public void loginAction(User user, HttpServletRequest request, HttpServletResponse response) {
String exceptionClassName = (String) request.getAttribute("shiroLoginFailure");
Throwable exObj = (Throwable) request.getAttribute("exObj"); String error = null;
LOGGER.info("client " + user.getName() + "登录..." + exceptionClassName); if (UnknownAccountException.class.getName().equals(exceptionClassName)) {
error = "未知用户错误";
} else if (IncorrectCredentialsException.class.getName().equals(exceptionClassName)) {
error = "用户名/password错误";
} else if (IncorrectCaptchaException.class.getName().equals(exceptionClassName)) {
error = "验证码错误";
} else if (exObj != null && exObj.getClass().getName().equals(IllegalArgumentException.class.getName())) {
error = exObj.getMessage(); // 缺少某个字段
} else if (exceptionClassName != null) {
error = "其它错误:" + exceptionClassName;
} // 输出 JSON
ResponseHelper rsp = new ResponseHelper(response);
boolean isOk = error == null;
String msg = error == null ? "登录成功!" : error + "详细原因:" + (exObj != null ? exObj.getMessage() : "N/A");
rsp.outputAction(isOk, msg);
}
可见,这个控制器好像“打杂”的,作用比較简单,像是一些善后的工作。
可是,为什么过滤器运行不起来?
遗憾的是,声明过滤器之后却运行不起来,不能正确跳转。
訪问这个 url。即使输入合法也跳到“登录”页面(也就是未登录的页面)。然后重复搜索相关资料和样例,得知表单是 AccessControlFilter 的子类,假设訪问了一定回 isAccessAllowed() 方法,试了下果然能够,表示是否同意訪问。然后我下意识地调用验证码的 executeLogin() 方法,也就是封装的 getSubject().login(token); 的方法。难道是这样调用控制器的吗?我覺得我这样的方法有点“简单粗暴”。框架不至于推荐这么做——后来的測试也证明此法不可行,报了一个两次 redirect 重定向的异常。——这肯定是我的方法不正确!
百搜不得其解下。我十分灰心。甚至想放弃,后来我想到把 Shiro 源代码和文档附加到 jar 中,看看里面的源代码和凝视。果然人家就说明须要 POST 请求。并且要指定登陆 URL(这个是 Action 接受请求的地址。假设不设置默认是 /login.jsp,这就是导致非常多人不能成功调用过滤器的原因!)才干调用过滤器,有条件限制的,并且根本不须要运行 isAccessAllowed() 方法。
后来又发现个小插曲,——偶尔登录成功后,在尝试登录是不会运行过滤器。因此 Shiro 定义是“一次运行的”。详细原理我就没有深究了仅仅是知道有这么一个机制。測试相应的方法也非常easy。把浏览器缓存清掉,令其 session 去掉。
另外表单 POST 字段的映射,FormAuthenticationFilter 也考虑到了,參见配置中的 <property name="usernameParam" value="name" />:
<bean id="captchaFormAuthenticationFilter"
class="com.ajaxjs.framework.user.captcha.CaptchaFormAuthenticationFilter">
<property name="usernameParam" value="name" />
<property name="loginUrl" value="/service/user/access/login" />
</bean>
这里的是 Spring 的配置方法。
小结一下 Shiro 认证
Shiro 就是这样,提供现成的组件方便我们调用,我们也能够从中了解其机制,——反正都是开源的。
过滤器是认证的第一板斧,是创建 token 的那一步。有了 token 才干运行 AuthenticatingRealm.doGetAuthenticationInfo(AuthenticationToken authcToken),最后到控制器。网上有些样例把创建 token 那一步写在控制器中。虽然做法有点简单粗暴,但道理还是一样的。
Shiro 学习应用(续)的更多相关文章
- Shiro学习笔记(5)——web集成
Web集成 shiro配置文件shiroini 界面 webxml最关键 Servlet 測试 基于 Basic 的拦截器身份验证 Web集成 大多数情况.web项目都会集成spring.shiro在 ...
- Shiro学习
Shiro学习资源 Shiro官网,http://shiro.apache.org/index.html 学习网站链接,http://blog.java1234.com/blog/articles/4 ...
- Apache Shiro学习-2-Apache Shiro Web Support
Apache Shiro Web Support 1. 配置 将 Shiro 整合到 Web 应用中的最简单方式是在 web.xml 的 Servlet ContextListener 和 Fil ...
- shiro学习笔记_0600_自定义realm实现授权
博客shiro学习笔记_0400_自定义Realm实现身份认证 介绍了认证,这里介绍授权. 1,仅仅通过配置文件来指定权限不够灵活且不方便.在实际的应用中大多数情况下都是将用户信息,角色信息,权限信息 ...
- Apache shiro学习总结
Apache shiro集群实现 (一) shiro入门介绍 Apache shiro集群实现 (二) shiro 的INI配置 Apache shiro集群实现 (三)shiro身份认证(Shiro ...
- Shiro学习笔记总结,附加" 身份认证 "源码案例(一)
Shiro学习笔记总结 内容介绍: 一.Shiro介绍 二.subject认证主体 三.身份认证流程 四.Realm & JDBC reaml介绍 五.Shiro.ini配置介绍 六.源码案例 ...
- SpringBoot+Shiro学习(七):Filter过滤器管理
SpringBoot+Shiro学习(七):Filter过滤器管理 Hiwayz 关注 0.5 2018.09.06 19:09* 字数 1070 阅读 5922评论 1喜欢 20 先从我们写的一个 ...
- Apache Shiro 学习记录5
本来这篇文章是想写从Factory加载ini配置到生成securityManager的过程的....但是貌似涉及的东西有点多...我学的又比较慢...很多类都来不及研究,我又怕等我后面的研究了前面的都 ...
- Apache Shiro 学习记录4
今天看了教程的第三章...是关于授权的......和以前一样.....自己也研究了下....我觉得看那篇教程怎么说呢.....总体上是为数不多的精品教程了吧....但是有些地方确实是讲的太少了.... ...
随机推荐
- Gym-101158J Cover the Polygon with Your Disk 计算几何 求动圆与多边形最大面积交
题面 题意:给出小于10个点形成的凸多边形 和一个半径为r 可以移动的圆 求圆心在何处的面积交最大,面积为多少 题解:三分套三分求出圆心位置,再用圆与多边形面积求交 #include<bits/ ...
- shp系列(二)——利用C++进行shp文件的读(打开)
1.各数据类型及其字节数 BYTE 1; char 1; short 2; int 4; double 8; 2.位序big和little及其转换 对于位序是big的 ...
- WEBGL学习笔记(七):实践练手1-飞行类小游戏之游戏控制
接上一节,游戏控制首先要解决的就是碰撞检测了 这里用到了学习笔记(三)射线检测的内容了 以鸟为射线原点,向前.上.下分别发射3个射线,射线的长度较短大概为10~30. 根据上一节场景的建设,我把y轴设 ...
- [Offer收割]编程练习赛39
公平分队 #pragma comment(linker, "/STACK:102400000,102400000") #include<stdio.h> #includ ...
- LeetCode 287. Find the Duplicate Number (python 判断环,时间复杂度O(n))
LeetCode 287. Find the Duplicate Number 暴力解法 时间 O(nlog(n)),空间O(n),按题目中Note"只用O(1)的空间",照理是过 ...
- 【SQL】SELECT 语句
1.1 SELECT基本语法: Select * |{[distinct]colum|expression [alias],…} from table; 1.2 查询当前用户所有在用的表及视图: HR ...
- junit使用第一弹
知识点——断言 断言是编写测试用例的核心实现方式,即期望值是多少,测试的结果是多少,以此来判断测试是否通过. 1. 断言核心方法 assertArrayEquals(expecteds, actual ...
- SQLite Tips
附加数据库 Attach database filename as database_name; 主数据库默认为 "Main", 在使用Attach命令时, 不能将database ...
- 使用PyQT编写界面程序
使用PyQT比QT好在,可以随时监测函数正确性,省去编译时间 ! 这是个不小的节省. 1. PyQt: 打开对话框 msgbox = QtGui.QMessageBox(self)# 我的语句是 ms ...
- Angular 通过constant(name,value),value(name,value)创建服务
区别: constant()可以将已经存在的变量值注册为服务,并将其注入到应用的其他部分中,他的value可以是值,也可以是对象.通过他来配置数据,也就是说可以在config里注入,但是他是不可以修改 ...