springboot redis shiro 实现 单点登录

大家好,我是烤鸭:

    今天给大家分享简单的单点登录的实现方式。

    环境及jar包:

    springboot     1.5.10   

    redis    2.9.0    (可以用tomcat的session,但是无法解决多个tomcat共享session的问题)

    shiro    1.4.0    

    lombok    1.16.10    (可选,编译的时候自动生成get,set,toString()方法)

登录方法:

    主要是用户完成登录,登录信息写在cookie,

    服务器端存缓存(redis)中。

LoginService:

[java]  view plain  copy
  1. package xxx.xxx.system.service.impl;  
  2.   
  3. import java.io.Serializable;  
  4. import java.util.HashMap;  
  5. import java.util.List;  
  6. import java.util.Map;  
  7.   
  8. import org.apache.shiro.SecurityUtils;  
  9. import org.apache.shiro.authc.AuthenticationException;  
  10. import org.apache.shiro.authc.UsernamePasswordToken;  
  11. import org.apache.shiro.subject.Subject;  
  12. import org.springframework.beans.factory.annotation.Autowired;  
  13. import org.springframework.stereotype.Service;  
  14.   
  15. import com.alibaba.fastjson.JSONObject;  
  16. import xxx.xxx.common.constant.IMsgEnum;  
  17. import xxx.xxx.common.resp.BaseMsgResp;  
  18. import xxx.xxx.common.utils.MD5Utils;  
  19. import xxx.xxx.entity.sys.UserDO;  
  20. import xxx.xxx.redis.cache.RedisManager;  
  21. import xxx.xxx.system.service.LoginService;  
  22. import xxx.xxx.system.service.UserService;  
  23.   
  24. @Service  
  25. public class LoginServiceImpl implements LoginService {  
  26.     @Autowired  
  27.     private UserService userService;  
  28.       
  29.     @Override  
  30.     public BaseMsgResp login(String username, String password) throws AuthenticationException{  
  31.         BaseMsgResp resp = new BaseMsgResp();  
  32.         password = MD5Utils.encrypt(username, password);  
  33.         UsernamePasswordToken token = new UsernamePasswordToken(username, password);  
  34.         Subject subject = SecurityUtils.getSubject();  
  35.         subject.login(token);  
  36.         //获取用户  
  37.         Map<String, Object> params = new HashMap<>();  
  38.         params.put("username", username);  
  39.         List<UserDO> list = userService.list(params);  
  40.         if(list.size() > 1) {  
  41.             resp = new BaseMsgResp(IMsgEnum.SYSTEM_ANOMALY.getMsgCode()+"",IMsgEnum.SYSTEM_ANOMALY.getMsgText());  
  42.             return resp;  
  43.         }  
  44.         //token  
  45.         Serializable id = subject.getSession().getId();  
  46.         //将token放入redis  
  47.         RedisManager manager = RedisManager.getRedisSingleton();  
  48.         manager.set("sys:login:user_token_"+id.toString(),list.get(0).getUserId()+"",60*30);  
  49.         //防止同一个账号同时登录  
  50.         manager.set("sys:user:id_"+list.get(0).getUserId(), id.toString(),60*30);  
  51.         //用户信息  
  52.         manager.set("sys:login:user_info_"+list.get(0).getUserId(), JSONObject.toJSONString(list.get(0)),60*30);  
  53.         return resp;  
  54.     }  
  55. }  

RedisManager.java

    用于redis初始化,和一些基本方法:

[java]  view plain  copy
  1. packagexxx.xxx.redis.cache;  
  2.   
  3.   
  4. import java.util.ArrayList;  
  5. import java.util.Iterator;  
  6. import java.util.List;  
  7.   
  8.   
  9. /** 
  10.  * @author  
  11.  * @version V1.0 
  12.  */  
  13.   
  14.   
  15. import java.util.Set;  
  16. import java.util.concurrent.TimeUnit;  
  17.   
  18.   
  19. import org.slf4j.Logger;  
  20. import org.slf4j.LoggerFactory;  
  21. import org.springframework.boot.context.properties.ConfigurationProperties;  
  22. import org.springframework.context.annotation.Configuration;  
  23.   
  24.   
  25. import com.alibaba.fastjson.JSONObject;  
  26. import xxx.xxx.redis.config.RedisConstant;  
  27.   
  28.   
  29. import lombok.Data;  
  30. import redis.clients.jedis.Jedis;  
  31. import redis.clients.jedis.JedisPool;  
  32. import redis.clients.jedis.JedisPoolConfig;  
  33.   
  34.   
  35. /** 
  36.  * 
  37.  */  
  38. @Configuration  
  39. @ConfigurationProperties(value="jedis.pool")  
  40. @Data  
  41. public class RedisManager {  
  42.     private final Logger logger = LoggerFactory.getLogger(this.getClass());  
  43.       
  44.     volatile static RedisManager redisSingleton;  
  45.       
  46.     private String host;  
  47.     private int port;  
  48.     private int expire;  
  49.     private int timeout;  
  50.     private String password = "";  
  51.     private static JedisPool jedisPool = null;  
  52.     //第几个仓库  
  53.     private String database;  
  54.       
  55.     public String getDatabase() {  
  56.         if(null == database || "".equals(database)) return "0";  
  57.         return database;  
  58.     }  
  59.   
  60.   
  61.     public void setDatabase(String database) {  
  62.         this.database = database;  
  63.     }  
  64.       
  65.     public RedisManager() {  
  66.   
  67.   
  68.     }  
  69.       
  70.   
  71.   
  72.     public static RedisManager getRedisSingleton() {  
  73.         if(redisSingleton == null) {  
  74.             synchronized (RedisManager.class) {  
  75.                 if(redisSingleton == nullreturn new RedisManager();  
  76.             }  
  77.         }  
  78.         return redisSingleton;  
  79.     }  
  80.   
  81.   
  82.     /** 
  83.      * 初始化方法 
  84.      */  
  85.     public void init() {  
  86.         if (jedisPool == null) {  
  87.             if (password != null && !"".equals(password)) {  
  88.                 jedisPool = new JedisPool(new JedisPoolConfig(), host, port, timeout, password);  
  89.             } else if (timeout != 0) {  
  90.                 jedisPool = new JedisPool(new JedisPoolConfig(), host, port, timeout);  
  91.             } else {  
  92.                 jedisPool = new JedisPool(new JedisPoolConfig(), host, port);  
  93.             }  
  94.   
  95.   
  96.         }  
  97.     }  
  98.   
  99.   
  100.   
  101.   
  102.     /** 
  103.      * 给Redis中Set集合中某个key值设值 
  104.      *  
  105.      * @param key 
  106.      * @param value 
  107.      */  
  108.     public void set(String key, String value) {  
  109.         Jedis jedis = null;  
  110.         try {  
  111.             jedis = jedisPool.getResource();  
  112.             jedis.select(Integer.parseInt(getDatabase()));  
  113.             jedis.set(key, value);  
  114.         } catch (Exception e) {  
  115.             logger.error("Jedis set 异常" + e.getMessage());  
  116.         } finally {  
  117.             if (jedis != null) {  
  118.                 jedis.close();  
  119.             }  
  120.         }  
  121.     }  
  122.   
  123.   
  124.     /** 
  125.      * 从Redis中Set集合中获取key对应value值 
  126.      *  
  127.      * @param key 
  128.      */  
  129.     public String get(String key) {  
  130.         Jedis jedis = null;  
  131.         try {  
  132.             jedis = jedisPool.getResource();  
  133.             jedis.select(Integer.parseInt(getDatabase()));  
  134.             return jedis.get(key);  
  135.         } catch (Exception e) {  
  136.             logger.error("Jedis get 异常" + e.getMessage());  
  137.             return null;  
  138.         } finally {  
  139.             if (jedis != null) {  
  140.                 jedis.close();  
  141.             }  
  142.         }  
  143.     }  
  144.   
  145.   
  146.   
  147.   
  148.   
  149.   
  150.     /** 
  151.      * 给Redis中Set集合中某个key值设值 
  152.      *  
  153.      * @param key 
  154.      * @param value 
  155.      */  
  156.     public void set(String key, String value, long time) {  
  157.         Jedis jedis = null;  
  158.         try {  
  159.             jedis = jedisPool.getResource();  
  160.             jedis.select(Integer.parseInt(getDatabase()));  
  161.             jedis.set(key, value);  
  162.             jedis.set(key, value, "XX""EX", time);  
  163.         } catch (Exception e) {  
  164.             logger.error("Jedis set 异常" + e.getMessage());  
  165.         } finally {  
  166.             if (jedis != null) {  
  167.                 jedis.close();  
  168.             }  
  169.         }  
  170.     }  
  171.   
  172.   
  173.       
  174.     /** 
  175.      *  
  176.     * @Title: expire  
  177.     * @Description: 设置指定key的生存时间    
  178.     * @return Long    成功 返回 1  
  179.     * @throws 
  180.      */  
  181.     public Long expire(String key,int seconds) {  
  182.         Jedis jedis = null;  
  183.         try {  
  184.             jedis = jedisPool.getResource();  
  185.             jedis.select(Integer.parseInt(getDatabase()));  
  186.             return jedis.expire(key, seconds);  
  187.         } catch (Exception e) {  
  188.             logger.error("Jedis expire 异常" + e.getMessage());  
  189.             return null;  
  190.         } finally {  
  191.             if (jedis != null) {  
  192.                 jedis.close();  
  193.             }  
  194.         }  
  195.   
  196.   
  197.     }  
  198.     /** 
  199.      *  
  200.     * @Title: exist  
  201.     * @Description: 判断key是否存在    
  202.     * @return Boolean    返回类型  
  203.     * @throws 
  204.      */  
  205.     public Boolean exist(String key) {  
  206.         Jedis jedis = null;  
  207.         try {  
  208.             jedis = jedisPool.getResource();  
  209.             jedis.select(Integer.parseInt(getDatabase()));  
  210.             return jedis.exists(key);  
  211.         } catch (Exception e) {  
  212.             logger.error("Jedis exist 异常" + e.getMessage());  
  213.             return null;  
  214.         } finally {  
  215.             if (jedis != null) {  
  216.                 jedis.close();  
  217.             }  
  218.         }  
  219.   
  220.   
  221.     }  
  222.       
  223. }  

LoginInterceptor:

    每次请求都会经过拦截器,校验用户是否登录或者多个浏览器(客户端)登录。

    如果用户已登录,用这次请求cookie中的token和缓存中的比较,如果不一致,就把之前缓存中的用户信息清空。

    这样之前的用户如果再次操作,就会跳转到登陆页面。

[java]  view plain  copy
  1. package xxx.xxx.system.interceptor;  
  2.   
  3. import java.io.IOException;  
  4. import java.io.Serializable;  
  5. import java.lang.ProcessBuilder.Redirect;  
  6. import java.util.HashMap;  
  7. import java.util.Map;  
  8.   
  9. import javax.servlet.ServletOutputStream;  
  10. import javax.servlet.http.Cookie;  
  11. import javax.servlet.http.HttpServletRequest;  
  12. import javax.servlet.http.HttpServletResponse;  
  13.   
  14. import org.apache.commons.lang3.StringUtils;  
  15. import org.apache.commons.logging.Log;  
  16. import org.apache.commons.logging.LogFactory;  
  17. import org.apache.shiro.SecurityUtils;  
  18. import org.apache.shiro.subject.Subject;  
  19. import org.springframework.beans.factory.annotation.Autowired;  
  20. import org.springframework.boot.autoconfigure.EnableAutoConfiguration;  
  21. import org.springframework.boot.autoconfigure.web.ErrorMvcAutoConfiguration;  
  22. import org.springframework.stereotype.Component;  
  23. import org.springframework.web.bind.annotation.RequestMethod;  
  24. import org.springframework.web.method.HandlerMethod;  
  25. import org.springframework.web.servlet.HandlerInterceptor;  
  26. import org.springframework.web.servlet.ModelAndView;  
  27.   
  28. import com.alibaba.fastjson.JSONObject;  
  29. import xxx.xxx.common.utils.LogUtils;  
  30. import xxx.xxx.redis.cache.RedisManager;  
  31. import xxx.xxx.system.utils.ShiroUtils;  
  32.   
  33. /** 
  34.  * ClassName: PlatformInterceptor date: 2015年12月30日 下午2:13:24 Description: 拦截器 
  35.  *  
  36.  * @author  
  37.  * @version 
  38.  * @since JDK 1.8 
  39.  */  
  40. @Component  
  41. @EnableAutoConfiguration(exclude = ErrorMvcAutoConfiguration.class)  
  42. public class LoginInterceptor implements HandlerInterceptor {  
  43.   
  44.     private static final Log logger = LogFactory.getLog(LoginInterceptor.class);  
  45.       
  46.     @Override  
  47.     public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) {  
  48.         try {  
  49.             logger.info(LogUtils.getRequestLog(request));  
  50.             //校验用户是否已经登录,如果登录过,将之前用户踢掉,同时更新缓存中用户信息  
  51.             Subject subject = SecurityUtils.getSubject();  
  52.             Serializable token = subject.getSession().getId();  
  53.             RedisManager redisManager = RedisManager.getRedisSingleton();  
  54.             //获取用户id  
  55.             String userId = redisManager.get("sys:login:user_token_"+token.toString());  
  56.             if(StringUtils.isNotBlank(userId)) {  
  57.                 String tokenPre = redisManager.get("sys:user:id_"+userId);  
  58.                 if(!token.equals(tokenPre)) {  
  59.                      //重定向到login.html  
  60.                     redirect(request, response);   
  61.                     return false;  
  62.                 }else {  
  63.                     Long expire = redisManager.ttl("sys:login:user_token_"+token.toString());  
  64.                     //过期时间小于1分钟的,更新token  
  65.                     if(expire < 1 * 60 * 1000) {  
  66.                         redisManager.expire("sys:login:user_token_"+token.toString(), 60*30);  
  67.                     }  
  68.                 }  
  69.             }else {  
  70.                 redirect(request, response);   
  71.                 return false;  
  72.             }  
  73.         } catch (Exception e) {  
  74.             logger.info("preHandle="+e.getMessage());  
  75.         }  
  76.         return true;  
  77.     }  
  78.   
  79.     @Override  
  80.     public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler,  
  81.             ModelAndView modelAndView) throws Exception {  
  82.   
  83.     }  
  84.   
  85.     @Override  
  86.     public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex)  
  87.             throws Exception {  
  88.     }  
  89.       //对于请求是ajax请求重定向问题的处理方法  
  90.     public void redirect(HttpServletRequest request, HttpServletResponse response) throws IOException{  
  91.         //获取当前请求的路径  
  92.         String basePath = request.getScheme() + "://" + request.getServerName() + ":"  + request.getServerPort()+request.getContextPath();  
  93. //        response.getOutputStream().write("账号在别处登录。".getBytes("UTF-8"));  
  94.         //如果request.getHeader("X-Requested-With") 返回的是"XMLHttpRequest"说明就是ajax请求,需要特殊处理 否则直接重定向就可以了  
  95.         if("XMLHttpRequest".equals(request.getHeader("X-Requested-With"))){  
  96.             //告诉ajax我是重定向  
  97.             response.setHeader("REDIRECT""REDIRECT");  
  98.             //告诉ajax我重定向的路径  
  99.             response.setHeader("CONTENTPATH", basePath+"/login");  
  100.             response.setStatus(HttpServletResponse.SC_FORBIDDEN);  
  101.         }else{  
  102.             response.sendRedirect(basePath + "/login");  
  103.         }  
  104.     }  
  105. }  

ShiroUtils.java

    shiro 工具类,获取subject对象。

[java]  view plain  copy
  1. package xxx.xxx.system.utils;  
  2.   
  3.   
  4. import java.security.Principal;  
  5. import java.util.Collection;  
  6. import java.util.List;  
  7.   
  8.   
  9. import org.apache.shiro.SecurityUtils;  
  10. import org.apache.shiro.session.Session;  
  11. import org.apache.shiro.session.mgt.eis.SessionDAO;  
  12. import org.apache.shiro.subject.Subject;  
  13. import org.springframework.beans.factory.annotation.Autowired;  
  14.   
  15.   
  16. import xxx.xxx.entity.sys.UserDO;  
  17.   
  18.   
  19. public class ShiroUtils {  
  20.     @Autowired  
  21.     private static SessionDAO sessionDAO;  
  22.   
  23.   
  24.     public static Subject getSubjct() {  
  25.         return SecurityUtils.getSubject();  
  26.     }  
  27.     public static UserDO getUser() {  
  28.         Object object = getSubjct().getPrincipal();  
  29.         return (UserDO)object;  
  30.     }  
  31.     public static Integer getUserId() {  
  32.         return getUser().getUserId();  
  33.     }  
  34.     public static void logout() {  
  35.         getSubjct().logout();  
  36.     }  
  37.   
  38.   
  39.     public static List<Principal> getPrinciples() {  
  40.         List<Principal> principals = null;  
  41.         Collection<Session> sessions = sessionDAO.getActiveSessions();  
  42.         return principals;  
  43.     }  
  44. }  

UserDO.java

    用户对象。

[java]  view plain  copy
  1. package xxx.xxx.entity.sys;  
  2.   
  3. import org.springframework.format.annotation.DateTimeFormat;  
  4.   
  5. import java.io.Serializable;  
  6. import java.util.Date;  
  7. import java.util.List;  
  8.   
  9. public class UserDO implements Serializable {  
  10.     private static final long serialVersionUID = 1L;  
  11.     //用户id  
  12.     private Integer userId;  
  13.     // 用户名  
  14.     private String username;  
  15.     // 用户真实姓名  
  16.     private String name;  
  17.     // 密码  
  18.     private String password;  
  19. }  

application.yml

[html]  view plain  copy
  1. jedis :  
  2.   pool :  
  3.     host : 127.0.0.1  
  4.     port : 9001  
  5.     password: abcd123  
  6.     maxTotal: 100  
  7.     maxIdle: 10  

  1.     maxWaitMillis : 100000  
  2. 转载地址:https://blog.csdn.net/Angry_Mills/article/details/80111780
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值