《Spring Security3》第三章第三部分翻译下(Remember me安全吗?)

Remember me是否安全?

对我们精心保护的站点来说,为了用户体验而添加的任何与安全相关的功能,都有增加安全风险的潜在可能。按照其默认方式,Remember me功能存在用户的cookie被拦截并被恶意用户重用的风险。下图展现了这种情况是如何发生的:

 


 

使用SSL(第四章进行讨论)以及其他的网络安全技术能缓解这种类型攻击的风险,但是要注意的是还有其他技术如跨站脚本攻击(XSS)能够窃取或损害一个remembered user session。为了照顾用户的易用性,我们不会愿意让用户的财产信息或个人信息因为remembered session的不合理使用而遭到篡改或窃取。

 

【尽管我们不会涉及恶意用户行为的细节,但是当你实现安全系统时,了解恶意用户所使用的攻击技术是很重要的。XSS是其中的一种技术,当然还有其他的很多种。强烈建议你了解OWASP Top Ten(http://www.owasp.org/index.php/Category:OWASP_Top_Ten_Project)作为一个入门列表并参考一本web应用安全参考书,里面介绍了各种的技术使用。】

 

平衡用户易用性和应用安全性的一种通用方法是识别出站点中与个人或敏感信息相关的功能点。确保这些功能点在进行授权校验时不仅要判断用户的角色,还要保证用户进行了完整的用户名和密码认证。这可以通过使用SpEL表达式语言的fullyAuthenticated伪属性来实现,关于授权规则的SpEL表达式语言我们在第二章中已经有所介绍。

Remember me认证与完整认证的在认证规则上的区别

我们将在随后的第五章:精确的访问控制中介绍高级的认证技术,但是,了解能够辨别认证session是否为remembered并以此建立访问规则也是很重要的。

 

我们可以设想一个使用remembered session登录的用户要查看和修改他的“wish list”。这与其他的客户在线站点很类似,并不会出现与用户信息或财务信息相关的风险(要注意的是每个站点各不相同,不能盲目的将这些规则应用与你的站点)。相反的,我们将会重点保护用户的账号以及订单功能。我们要确保即使是remembered的用户,如果试图访问账号信息或定购产品,都需要对他们进行认证。以下为我们如何设置授权规则:

 

 <intercept-url pattern="/login.do" access="permitAll"/>

Xml代码   收藏代码
  1. <intercept-url pattern="/account/*.do"  
  2.    access="hasRole('ROLE_USER') and fullyAuthenticated"/>  
  3. <intercept-url pattern="/*" access="hasRole('ROLE_USER')"/>  

 已经存在的登录页和ROLE_USER设置没有变化。但我们添加了一条规则,要求用户具有GrantedAuthority的ROLE_USER角色同时还要求用户被完全的认证,即这个认证的session确实是通过提供用户名、密码或等同的凭证来进行认证的。注意这里的SpEL逻辑操作语法——在SpEL中,使用and,or以及not作为逻辑操作符。这是SpEL的设计者充分考虑的结果,因为&&操作符在XML中很难被使用。

 

如果你在应用中尝试运行,如果以remember me功能登录并试图访问“My Account”链接,你将会得到一个403访问拒绝的提示,这说明这个地址已经被适当地保护了。出现错误界面是因为我们应用的配置还是使用默认的AccessDeniedHandler,这个类负责捕获和响应AccessDeniedException的信息。我们将会在第六章学习AccessDeniedException怎样被处理时,自定义这个行为。

 

【不使用表达式来实现完全认证的检查。如果你的应用不使用SpEL表达式来进行访问控制声明,你可以通过使用IS_AUTHENTICATED_FULLY访问规则来检查用户是不是进行了完整的认证(如:access=" IS_AUTHENTICATED_FULLY")。但要注意的是,这种标准的角色设置声明并没有SpEL那样强的表现力,所以如果要处理复杂的boolean表达式的时候,可能会比较困难。】

 

错误处理尚没有添加,但是你可以看到通过这种方式将remember me的易用性与更高层次的安全性结合了起来,用户访问敏感的信息时就会被要求提供完整的凭证信息。

构建一个关联IP的Remember me Service

有一种让remember me功能更安全的方式就是将用户的IP地址绑定到cookie的内容上。让我们通过一个例子来描述怎样构建RememberMeServices的实现类来完成这个功能。

 

基本的实现方式是扩展o.s.s.web.authentication.rememberme.TokenBasedRememberMeServices基类,以添加请求者的IP地址到cookie本身和其他的MD5哈希元素中。

 

扩展这个基类涉及到重写两个主要方法,并重写或实现几个小的帮助方法。还有一个要注意的是我们需要临时存储HttpServletRequest(将使用它来得到用户的IP地址)到一个ThreadLocal中,因为基类中的一些方法并没有将HttpServletRequest作为一个参数。

扩展TokenBasedRememberMeServices

首先,我们要扩展TokenBasedRememberMeServices类并重写父类的特定行为。尽管父类是非常易于重写,但是我们不想去重复一些重要的处理流程,所以能使这个类非常简明却有点不好理解。在com.packtpub.springsecurity.security包下创建这个类:

 

Java代码   收藏代码
  1. public class IPTokenBasedRememberMeServices extends  
  2.     TokenBasedRememberMeServices {  

 

 还有一些简单的方法来设置和获取ThreadLocal HttpServletRequest:

 

Java代码   收藏代码
  1. private static final ThreadLocal<HttpServletRequest> requestHolder =   
  2. new ThreadLocal<HttpServletRequest>();  
  3. public HttpServletRequest getContext() {  
  4.     return requestHolder.get();  
  5. }  
  6. public void setContext(HttpServletRequest context) {  
  7.     requestHolder.set(context);  
  8. }  
 

我们还需要添加一个工具方法以从HttpServletRequest中获取IP地址:

 

Java代码   收藏代码
  1. protected String getUserIPAddress(HttpServletRequest request) {  
  2.   return request.getRemoteAddr();  
  3. }  

 我们要重写的第一个有趣的方法是onLoginSuccess,它用来为remember me处理设置cookie的值。在这个方法中,我们需要设置ThreadLocal并在完成处理后将其清除。需要记住的是父类方法的处理流程——收集用户的所有认证请求信息并将其合成到cookie中。

 

Java代码   收藏代码
  1. @Override  
  2. public void onLoginSuccess(HttpServletRequest request,  
  3.     HttpServletResponse response,  
  4.     Authentication successfulAuthentication) {  
  5.   try  
  6.   {  
  7.     setContext(request);  
  8.     super.onLoginSuccess(request, response, successfulAuthentication  
  9.   }  
  10.   finally  
  11.   {    setContext(null);  
  12.   }  
  13. }  

 父类的onLoginSuccess方法将会触发makeTokenSignature方法来创建认证凭证的MD5哈希值。我们将要重写此方法,以实现从request中获取IP地址并使用Spring框架的一个工具类编码要返回的cookie值。(这个方法在进行remember me校验时还会被调用到,以判断前台传递过来的cookie值与后台根据用户名、密码、IP地址等信息生成的MD5值是否一致。——译者注)

 

Java代码   收藏代码
  1. @Override  
  2. protected String makeTokenSignature(long tokenExpiryTime,   
  3.     String username, String password) {  
  4.     return DigestUtils.md5DigestAsHex((username + ":" +   
  5. tokenExpiryTime + ":" + password + ":" + getKey() + ":" + getUserIPAdd  
  6. ress(getContext())).getBytes());   
  7. }  

 与之类似的,我们还重写了setCookie方法以添加包含IP地址的附加编码信息:

 

Java代码   收藏代码
  1. @Override  
  2. protected void setCookie(String[] tokens, int maxAge,  
  3.   HttpServletRequest request, HttpServletResponse response) {  
  4.   // append the IP adddress to the cookie  
  5.   String[] tokensWithIPAddress =   
  6.       Arrays.copyOf(tokens, tokens.length+1);  
  7.   tokensWithIPAddress[tokensWithIPAddress.length-1] =   
  8.       getUserIPAddress(request);  
  9.   super.setCookie(tokensWithIPAddress, maxAge,   
  10.       request, response);  
  11. }  

 这就得到了生成新cookie所有需要的信息。

 

最后,我们要重写processAutoLoginCookie方法,它用来校验用户端提供的remember me cookie的内容。父类已经为我们解决了大部分有意思的工作,但是,为了避免调用父类冗长的代码,我们在调用它之前先进行了一次IP地址的校验。

 

 

Java代码   收藏代码
  1. @Override  
  2. protected UserDetails processAutoLoginCookie(  
  3.   String[] cookieTokens,  
  4.   HttpServletRequest request, HttpServletResponse response)   
  5. {  
  6.   try  
  7.   {  
  8.     setContext(request);  
  9.   // take off the last token  
  10.     String ipAddressToken = cookieTokens[cookieTokens.length-1];  
  11.     if(!getUserIPAddress(request).equals(ipAddressToken))  
  12.     {  
  13.           throw new InvalidCookieException("Cookie IP Address did not   
  14. contain a matching IP (contained '" + ipAddressToken + "')");  
  15.     }  
  16.         
  17.     return super.processAutoLoginCookie(Arrays.copyOf(cookieTokens,   
  18. cookieTokens.length-1), request, response);  
  19.   }  
  20.   finally  
  21.   {  
  22.     setContext(null);  
  23.   }  
  24. }  

 我们的自定义的RememberMeServices编码已经完成了。现在我们要进行一些微小的配置。这个类的完整源代码(包括附加的注释)都在本章的源码中能够找到。

 

配置自定义的RememberMeServices

配置自定义的RememberMeServices实现需要两步来完成。第一步是修改dogstore-base.xml Spring配置文件,以添加我们刚刚完成类的SpringBean声明:

 

 

Xml代码   收藏代码
  1. <bean class="com.packtpub.springsecurity.security.IPTokenBasedRememberMeServices" id="ipTokenBasedRememberMeServicesBean">  
  2.   <property name="key"><value>jbcpPetStore</value></property>  
  3.   <property name="userDetailsService" ref="userService"/>  
  4. </bean>  

 第二个要进行的修改是Spring Security的XML配置文件。修改<remember-me>元素来引用自定义的Spring Bean,如下所示:

 

Xml代码   收藏代码
  1. <remember-me key="jbcpPetStore"   
  2.    services-ref="ipTokenBasedRememberMeServicesBean"/>  

 最后为<user-service>声明添加一个id属性,如果它还没有添加的话:

 

Xml代码   收藏代码
  1. <user-service  id="userService">  

 重启web应用,你将能看到新的IP过滤功能已经生效了。

 

因为remember me cookie是Base64编码的,我们能够使用一个Base64解码的工具得到cookie的值以证实我们的新增功能是否生效。如果我们这样做的话,我们能够看到一个名为SPRING_SECURITY_REMEMBER_ME_COOKIE的cookie的内容大致如下所示:

guest:1251695034322:776f8ad44034f77d13218a5c431b7b34:127.0.0.1

 

正如我们所料,你能够看到IP地址确实存在于cookie的结尾处。在IP地址之前,你还能够分别看到用户名、时间戳以及MD5的哈希值。

 

【调试remember me cookie。在尝试调试remember me功能时,会有两个难点。第一个就是得到cookie的值本身!Spring Security并没有提供记录我们设置的cookie值的日志级别。我们推荐使用基于浏览器的工具如Mozilla Firefox下的Chris Pederick's Web Developer插件(http://chrispederick.com/work/web-developer/)。基于浏览器的开发工具一般允许查看(甚至编辑)cookie的值。第二个困难(相对来说较小)就是解码cookie的值。你能使用在线或离线的Base64解码工具来对cookie的值进行解码(需要记住的是添加一个等号符(=)结尾,以使其成为一个合法的Base64编码值)。】

 

如果用户是在一个共享的或负载均衡的网络设施下,如multi-WAN公司环境,基于IP的remember me tokens可能会出现问题。但是在大多数场景下,添加IP地址到remember me功能能够为用户提供功能更强、更好的安全层。

 

 

自定义Remember me的签名

好奇的读者可能会关心remember me form的checkbox名(_spring_security_remember_me)以及cookie的名(SPRING_SECURITY_REMEMBER_ME_COOKIE),是否能够修改。<remember-me>声明是不支持这种扩展性的,但是现在我们作为一个Spring Bean声明了自己的RememberMeServices实现,那我们能够定义更多的属性来改变checkbox和cookie的名字:

 

 

Xml代码   收藏代码
  1. <bean class="com.packtpub.springsecurity.web.custom.  
  2. IPTokenBasedRememberMeServices" id="ipTokenBasedRememberMeServicesBean">  
  3. <property name="key"><value>jbcpPetStore</value></property>  
  4.   <property name="userDetailsService" ref="userService"/>  
  5.   <property name="parameter" value="_remember_me"/>  
  6.   <property name="cookieName" value="REMEMBER_ME"/>  
  7. </bean>  
 

 

不要忘记的是,还需要修改login.jsp页面中的checkbox form域以与我们声明的parameter值相匹配。我们建议你进行一下实验以确保理解这些设置之间的关联。

(如果想更好的理解本章节内容,建议阅读一下Spring Security的源码——译者注)

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
目标检测(Object Detection)是计算机视觉领域的一个核心问题,其主要任务是找出图像中所有感兴趣的目标(物体),并确定它们的类别和位置。以下是对目标检测的详细阐述: 一、基本概念 目标检测的任务是解决“在哪里?是什么?”的问题,即定位出图像中目标的位置并识别出目标的类别。由于各类物体具有不同的外观、形状和姿态,加上成像时光照、遮挡等因素的干扰,目标检测一直是计算机视觉领域最具挑战性的任务之一。 二、核心问题 目标检测涉及以下几个核心问题: 分类问题:判断图像中的目标属于哪个类别。 定位问题:确定目标在图像中的具体位置。 大小问题:目标可能具有不同的大小。 形状问题:目标可能具有不同的形状。 三、算法分类 基于深度学习的目标检测算法主要分为两大类: Two-stage算法:先进行区域生成(Region Proposal),生成有可能包含待检物体的预选框(Region Proposal),再通过卷积神经网络进行样本分类。常见的Two-stage算法包括R-CNN、Fast R-CNN、Faster R-CNN等。 One-stage算法:不用生成区域提议,直接在网络中提取特征来预测物体分类和位置。常见的One-stage算法包括YOLO系列(YOLOv1、YOLOv2、YOLOv3、YOLOv4、YOLOv5等)、SSD和RetinaNet等。 四、算法原理 以YOLO系列为例,YOLO将目标检测视为回归问题,将输入图像一次性划分为多个区域,直接在输出层预测边界框和类别概率。YOLO采用卷积网络来提取特征,使用全连接层来得到预测值。其网络结构通常包含多个卷积层和全连接层,通过卷积层提取图像特征,通过全连接层输出预测结果。 五、应用领域 目标检测技术已经广泛应用于各个领域,为人们的生活带来了极大的便利。以下是一些主要的应用领域: 安全监控:在商场、银行
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值