若依项目如何实现一个账户只能一个人登录(即,限制账户不允许多终端登录)

目录

一、不分离版

在application.yml设置maxSession为1即可。修改shiro的配置

二、分离版springboot+vue

1、application.yml新增一个配置soloLogin用于限制多终端同时登录。修改token的配置。

2、Constants.java新增一个常量LOGIN_USERID_KEY公用。

3、调整TokenService.java,存储&刷新缓存用户编号信息

 4、自定义退出处理类LogoutSuccessHandlerImpl.java清除缓存方法添加用户编号。

5、登录方法SysLoginService.java,验证如果用户不允许多终端同时登录,清除缓存信息。


一、不分离版

application.yml设置maxSession为1即可。

修改shiro的配置

shiro:
  session:
    # 同一个用户最大会话数,比如2的意思是同一个账号允许最多同时两个人登录(默认-1不限制)
    maxSession: 1
    # 踢出之前登录的/之后登录的用户,默认踢出之前登录的用户
    kickoutAfter: false

二、分离版springboot+vue

1、application.yml新增一个配置soloLogin用于限制多终端同时登录。

修改token的配置。

token:
    # 是否允许账户多终端同时登录(true允许 false不允许)
    soloLogin: false

2、Constants.java新增一个常量LOGIN_USERID_KEY公用。

路径:com.zhdj.common.constant.Constants.java。

/**
 * 登录用户编号 redis key
 */
public static final String LOGIN_USERID_KEY = "login_userid:";

3、调整TokenService.java,存储&刷新缓存用户编号信息

路径:com.zhdj.framework.web.service.TokenService.java

// 是否允许账户多终端同时登录(true允许 false不允许)
@Value("${token.soloLogin}")
private boolean soloLogin;

/**
 * 删除用户身份信息
 */
public void delLoginUser(String token, Long userId)
{
	if (StringUtils.isNotEmpty(token))
	{
		String userKey = getTokenKey(token);
		redisCache.deleteObject(userKey);
	}
    //在原有代码上添加下面的代码
	if (!soloLogin && StringUtils.isNotNull(userId))
	{
		String userIdKey = getUserIdKey(userId);
		redisCache.deleteObject(userIdKey);
	}
}

/**
 * 刷新令牌有效期
 * 
 * @param loginUser 登录信息
 */
public void refreshToken(LoginUser loginUser)
{
	loginUser.setLoginTime(System.currentTimeMillis());
	loginUser.setExpireTime(loginUser.getLoginTime() + expireTime * MILLIS_MINUTE);
	// 根据uuid将loginUser缓存
	String userKey = getTokenKey(loginUser.getToken());
	redisCache.setCacheObject(userKey, loginUser, expireTime, TimeUnit.MINUTES);
	//在原有代码上添加下面的代码
    if (!soloLogin)
	{
		// 缓存用户唯一标识,防止同一帐号,同时登录
		String userIdKey = getUserIdKey(loginUser.getUser().getUserId());
		redisCache.setCacheObject(userIdKey, userKey, expireTime, TimeUnit.MINUTES);
	}
}

//添加下面的代码
private String getUserIdKey(Long userId)
{
	return Constants.LOGIN_USERID_KEY + userId;
}

 4、自定义退出处理类LogoutSuccessHandlerImpl.java清除缓存方法添加用户编号。

路径:com.zhdj.framework.security.handle.LogoutSuccessHandlerImpl.java

// 删除用户缓存记录
// 在原代码参数中添加用户编号:loginUser.getUser().getUserId()
tokenService.delLoginUser(loginUser.getToken(), loginUser.getUser().getUserId());

5、登录方法SysLoginService.java,验证如果用户不允许多终端同时登录,清除缓存信息。

路径:com.zhdj.framework.web.service.SysLoginService.java

// 是否允许账户多终端同时登录(true允许 false不允许)
    @Value("${token.soloLogin}")
    private boolean soloLogin;

    /**
     * 登录验证
     * 
     * @param username 用户名
     * @param password 密码
     * @param code 验证码
     * @param uuid 唯一标识
     * @return 结果
     */
    public String login(String username, String password, String code, String uuid)
    {
        boolean captchaEnabled = configService.selectCaptchaEnabled();
        // 验证码开关
        if (captchaEnabled)
        {
            validateCaptcha(username, code, uuid);
        }
        // 用户验证
        Authentication authentication = null;
        try
        {
            UsernamePasswordAuthenticationToken authenticationToken = new UsernamePasswordAuthenticationToken(username, password);
            AuthenticationContextHolder.setContext(authenticationToken);
            // 该方法会去调用UserDetailsServiceImpl.loadUserByUsername
            authentication = authenticationManager.authenticate(authenticationToken);
        }
        catch (Exception e)
        {
            if (e instanceof BadCredentialsException)
            {
                AsyncManager.me().execute(AsyncFactory.recordLogininfor(username, Constants.LOGIN_FAIL, MessageUtils.message("user.password.not.match")));
                throw new UserPasswordNotMatchException();
            }
            else
            {
                AsyncManager.me().execute(AsyncFactory.recordLogininfor(username, Constants.LOGIN_FAIL, e.getMessage()));
                throw new ServiceException(e.getMessage());
            }
        }
        finally
        {
            AuthenticationContextHolder.clearContext();
        }
        AsyncManager.me().execute(AsyncFactory.recordLogininfor(username, Constants.LOGIN_SUCCESS, MessageUtils.message("user.login.success")));
        LoginUser loginUser = (LoginUser) authentication.getPrincipal();
        recordLoginInfo(loginUser.getUserId());
        // 添加到这里
        // 判断是否允许账户多终端同时登录
        if (!soloLogin)
        {
            // 如果用户不允许多终端同时登录,清除缓存信息
            String userIdKey = Constants.LOGIN_USERID_KEY + loginUser.getUser().getUserId();
            String userKey = redisCache.getCacheObject(userIdKey);
            if (StringUtils.isNotEmpty(userKey))
            {
                redisCache.deleteObject(userIdKey);
                redisCache.deleteObject(userKey);
            }
        }
        // 生成token
        return tokenService.createToken(loginUser);
    }

 参考资料:若依框架官方文档

  • 5
    点赞
  • 19
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
实现账户只能一个登录的功能,可以使用 Flask 的 session 和 Redis 等第三方库来实现。 具体实现步骤如下: 1. 在用户登录时,将用户的唯一标识(如用户 ID)存储到 session 中,并将该唯一标识作为 key,当前 session ID 作为 value,存储到 Redis 中。如果该 key 已经存在,说明该用户已经登录,需要先将该用户的 session ID 从 Redis 中删除,再存储新的 session ID。 2. 在用户每次请求时,先从 session 中获取用户的唯一标识,然后根据该唯一标识从 Redis 中获取当前 session ID。如果当前 session ID 与用户的 session ID 不一致,说明该用户已经在其他地方登录了,需要强制退出当前登录。 3. 在用户退出登录时,需要从 Redis 中删除该用户的 session ID。 下面是一个简单的实现示例: ```python from flask import Flask, session import uuid import redis app = Flask(__name__) app.secret_key = 'my-secret-key' redis_client = redis.Redis(host='localhost', port=6379, db=0) @app.route('/login/<user_id>') def login(user_id): # 先从 Redis 中查找该用户是否已经登录 session_id = redis_client.get(user_id) if session_id: # 强制退出已经登录的用户 redis_client.delete(session_id) session.pop('_user_id', None) # 将用户的唯一标识存储到 session 中 session['_user_id'] = user_id # 将该唯一标识和当前 session ID 存储到 Redis 中 redis_client.set(user_id, session.sid) return 'Logged in successfully' @app.route('/logout') def logout(): # 从 Redis 中删除当前用户的 session ID user_id = session.get('_user_id') if user_id: redis_client.delete(user_id) session.pop('_user_id', None) return 'Logged out successfully' if __name__ == '__main__': app.run() ``` 在上面的示例中,我们首先需要设置 Flask 应用程序的 secret_key,这个 key 用于保护 session 数据的安全性。然后在登录时,我们先从 Redis 中查找该用户是否已经登录,如果已经登录则强制退出;否则将用户的唯一标识存储到 session 中,并将该唯一标识和当前 session ID 存储到 Redis 中。在退出登录时,我们从 Redis 中删除该用户的 session ID。 需要注意的是,该实现方式仅能保证同一个账户只能一个地方登录,但不能保证同一时间只能一个账户登录。如果需要实现同一时间只能一个账户登录的功能,可以考虑使用锁机制或者其他的并发控制方法。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值