【Shiro】7、Shiro实现控制用户并发登录并踢人下线(下)

简介: 在传统的项目中,同一账户是允许多人同时登录在线的,有的使用场景恰恰是不允许多人同时在线的,那么我们可以通过 Shiro 来控制并发登录,并实现后登录的用户,挤掉前面登录的用户
  • 2、注册过滤器
/**
* 并发登录控制
*
* @return
*/
@Bean
public KickoutSessionFilter kickoutSessionControlFilter() {
   KickoutSessionFilter kickoutSessionControlFilter = new KickoutSessionFilter();
   // 用于根据会话ID,获取会话进行踢出操作的;
   kickoutSessionControlFilter.setSessionManager(sessionManager());
   // 使用cacheManager获取相应的cache来缓存用户登录的会话;用于保存用户—会话之间的关系的;
   kickoutSessionControlFilter.setCacheManager(ehCacheManager());
   // 是否踢出后来登录的,默认是false;即后者登录的用户踢出前者登录的用户;
   kickoutSessionControlFilter.setKickoutAfter(false);
   // 同一个用户最大的会话数,默认1;比如2的意思是同一个用户允许最多同时两个人登录;
   kickoutSessionControlFilter.setMaxSession(1);
   // 被踢出后重定向到的地址;
   kickoutSessionControlFilter.setKickoutUrl("/login?kickout=1");
   return kickoutSessionControlFilter;
}

其中用到的 session 管理器,和 ehcache 管理器在之前的博客中都有讲到,本次不再赘述

我们将踢出的用户重定向到登录界面,并携带参数 kickout

  • 3、注入自定义过滤器
/**
 * 地址过滤器
 *
 * @param securityManager
 * @return
 */
@Bean
public ShiroFilterFactoryBean shiroFilterFactoryBean(SecurityManager securityManager) {
    ShiroFilterFactoryBean shiroFilterFactoryBean = new ShiroFilterFactoryBean();
    // 设置securityManager
    shiroFilterFactoryBean.setSecurityManager(securityManager);
    // 设置登录url
    shiroFilterFactoryBean.setLoginUrl("/login");
    // 设置主页url
    shiroFilterFactoryBean.setSuccessUrl("/");
    // 设置未授权的url
    shiroFilterFactoryBean.setUnauthorizedUrl("/403");
    // 自定义拦截器限制并发人数
    LinkedHashMap<String, Filter> filtersMap = new LinkedHashMap<>();
    // 限制同一帐号同时在线的个数
    filtersMap.put("kickout", kickoutSessionControlFilter());
    shiroFilterFactoryBean.setFilters(filtersMap);
    Map<String, String> filterChainDefinitionMap = new LinkedHashMap<>();
    // 注销登录
    filterChainDefinitionMap.put("/loginOut", "logout");
    // 开放登录接口
    filterChainDefinitionMap.put("/doLogin", "anon");
    // 开放获取登录验证码接口
    filterChainDefinitionMap.put("/kaptcha/**", "anon");
    // 开放Api接口
    filterChainDefinitionMap.put("/api/**", "anon");
    // 开放微信接口
    filterChainDefinitionMap.put("/weixin/**", "anon");
    // 开放websocket接口
    filterChainDefinitionMap.put("/websocket/**", "anon");
    // 开放接口文档
    filterChainDefinitionMap.put("/doc.html", "anon");
    filterChainDefinitionMap.put("/service-worker.js", "anon");
    filterChainDefinitionMap.put("/swagger-resources/**", "anon");
    filterChainDefinitionMap.put("/webjars/**", "anon");
    filterChainDefinitionMap.put("/v2/**", "anon");
    // 开放静态资源
    filterChainDefinitionMap.put("/css/**", "anon");
    filterChainDefinitionMap.put("/img/**", "anon");
    filterChainDefinitionMap.put("/js/**", "anon");
    filterChainDefinitionMap.put("/layui/**", "anon");
    filterChainDefinitionMap.put("/layuimini/**", "anon");
    filterChainDefinitionMap.put("/module/**", "anon");
    filterChainDefinitionMap.put("/upload/**", "anon");
    // 其余url全部拦截,必须放在最后
    filterChainDefinitionMap.put("/**", "kickout,user");
    shiroFilterFactoryBean.setFilterChainDefinitionMap(filterChainDefinitionMap);
    return shiroFilterFactoryBean;
}

我们在 ShiroFilterFactoryBean 对象中注入了自定义过滤器,并在最后的地址拦截规则中增加了 kickout,即使我们开启了记住登录功能,该用户也会被踢下线

  • 4、提示信息

我们在登录页面中,需要获取地址中是否有 kickout 参数

// 是否被挤下线
if(location.href.indexOf("kickout") > 0){
    setTimeout(function () {
        layNotify.notice({
            title: "登录提示",
            type: "error",
            message: '您的账户已在另一台设备上登录,如非本人操作,请立即修改密码!'
        });
    }, 1000)
}

这样后面登录的用户就会挤掉前面登录用户,导致前面登录的用户被踢下线了

目录
相关文章
|
3月前
|
缓存 算法 Java
Shiro【散列算法、Shiro会话、退出登录 、权限表设计、注解配置鉴权 】(五)-全面详解(学习总结---从入门到深化)
Shiro【散列算法、Shiro会话、退出登录 、权限表设计、注解配置鉴权 】(五)-全面详解(学习总结---从入门到深化)
52 0
Shiro【散列算法、Shiro会话、退出登录 、权限表设计、注解配置鉴权 】(五)-全面详解(学习总结---从入门到深化)
|
4月前
|
算法 Java BI
Shiro【散列算法、Shiro会话、退出登录 、权限表设计、注解配置鉴权 】(五)-全面详解(学习总结---从入门到深化)(上)
Shiro【散列算法、Shiro会话、退出登录 、权限表设计、注解配置鉴权 】(五)-全面详解(学习总结---从入门到深化)
35 0
|
4月前
|
存储 缓存 算法
Shiro【散列算法、Shiro会话、退出登录 、权限表设计、注解配置鉴权 】(五)-全面详解(学习总结---从入门到深化)(下)
Shiro【散列算法、Shiro会话、退出登录 、权限表设计、注解配置鉴权 】(五)-全面详解(学习总结---从入门到深化)
27 0
|
10月前
Shiro用户鉴权框架 子线程获取不到用户信息问题解决
Shiro用户鉴权框架 子线程获取不到用户信息问题解决
|
12月前
|
存储 JSON 前端开发
基于Springboot外卖系统04:后台系统用户登录+登出功能
如果前端与后台接口之间不存在跨域问题,那么推荐使用cookie和session来记录登录状态。如果前端与服务器接口之间存在跨域问题,那么就要使用token的方式来维持登录状态。
188 0
|
安全 jenkins 持续交付
Jenkins 开启用户注册机制及用户权限设置
Jenkins 开启用户注册机制及用户权限设置
293 0
|
存储 缓存 数据安全/隐私保护
Jasny SSO是如何处理用户会话的?底层原理是什么?
Jasny SSO是如何处理用户会话的?底层原理是什么?
|
存储
【Shiro】7、Shiro实现控制用户并发登录并踢人下线(上)
在传统的项目中,同一账户是允许多人同时登录在线的,有的使用场景恰恰是不允许多人同时在线的,那么我们可以通过 Shiro 来控制并发登录,并实现后登录的用户,挤掉前面登录的用户
270 0
|
缓存 安全 数据库
shiro登录认证过程
shiro登录认证过程
shiro登录认证过程
|
关系型数据库 MySQL 测试技术
菜鸟学习shiro之用数据库作为Realm的认证源实现登录,角色身份的判定和权限验证3
Maven的的依赖和第一篇是一样的副本直接用 上两篇讲的无论是内部类的封装领域还是配置文件的认证源,这篇开始讲最关心的数据库作为境界的认证源 这里使用的是四郎给我们提供的另一个内置的类JdbcRealm这个是连接数据库的一个内部类,话不多说了,具体的代码如下: import com.
1583 0

热门文章

最新文章