设为首页 收藏本站
查看: 1105|回复: 0

[经验分享] Shiro经过Redis管理会话实现集群(转载)

[复制链接]

尚未签到

发表于 2017-12-20 17:48:23 | 显示全部楼层 |阅读模式
  

package com.gqshao.authentication.controller;  

  
import com.gqshao.authentication.dao.CachingShiroSessionDao;
  
import org.apache.shiro.session.Session;
  
import org.springframework.beans.factory.annotation.Autowired;
  
import org.springframework.stereotype.Controller;
  
import org.springframework.web.bind.annotation.RequestMapping;
  
import org.springframework.web.bind.annotation.ResponseBody;
  

  
import java.io.Serializable;
  
import java.util.Collection;
  

  
@Controller
  
@RequestMapping("/session")

  
public>  

  @Autowired
  private CachingShiroSessionDao sessionDao;
  

  @RequestMapping("/active")
  @ResponseBody
  public Collection<Session> getActiveSessions() {
  return sessionDao.getActiveSessions();
  }
  

  @RequestMapping("/read")
  @ResponseBody
  public Session readSession(Serializable sessionId) {
  return sessionDao.doReadSessionWithoutExpire(sessionId);
  }
  
}
  

  


七.集群情况下的改造
  1.问题上面启用了Redis中央缓存、EhCache本地JVM缓存,AuthorizingRealm的doGetAuthenticationInfo登陆认证方法返回的AuthenticationInfo,默认情况下会被保存到Session的Attribute下面两个字段中
  

org.apache.shiro.subject.support.DefaultSubjectContext.PRINCIPALS_SESSION_KEY 保存 principal  
org.apache.shiro.subject.support.DefaultSubjectContext.AUTHENTICATED_SESSION_KEY 保存 boolean是否登陆
  

  然后在每次请求过程中,在ShiroFilter中组装Subject时,读取Session中这两个字段
  现在的问题是Session被缓存到本地JVM堆中,也就是说服务器A登陆,无法修改服务器B的EhCache中Session属性,导致服务器B没有登陆。
  处理方法有很多思路,比如重写CachingSessionDAO,readSession如果没有这两个属性就不缓存(没登陆就不缓存),或者cache的session没有这两个属性就调用自己实现的doReadSession方法从Redis中重读一下。
  

/**  * 重写CachingSessionDAO中readSession方法,如果Session中没有登陆信息就调用doReadSession方法从Redis中重读
  */
  
@Override
  
public Session readSession(Serializable sessionId) throws UnknownSessionException {
  Session session = getCachedSession(sessionId);
  if (session == null
  || session.getAttribute(DefaultSubjectContext.PRINCIPALS_SESSION_KEY) == null) {
  session = this.doReadSession(sessionId);
  if (session == null) {

  throw new UnknownSessionException("There is no session with>  } else {
  // 缓存
  cache(session, session.getId());
  }
  }
  return session;
  
}
  

  2.如果需要保持各个服务器Session是完全同步的,可以通过Redis消息订阅/发布功能,再调用SessionDao中实现了删除Session本地缓存的方法

运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-426153-1-1.html 上篇帖子: redis cluster集群理解 下篇帖子: redis性能优化之redis.cnf配置文件
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表