• 企业400电话
  • 微网小程序
  • AI电话机器人
  • 电商代运营
  • 全 部 栏 目

    企业400电话 网络优化推广 AI电话机器人 呼叫中心 网站建设 商标✡知产 微网小程序 电商运营 彩铃•短信 增值拓展业务
    redis 实现登陆次数限制的思路详解

    title: redis-login-limitation 

    利用 redis 实现登陆次数限制, 注解 + aop, 核心代码很简单.

    基本思路

    比如希望达到的要求是这样: 在 1min 内登陆异常次数达到5次, 锁定该用户 1h

    那么登陆请求的参数中, 会有一个参数唯一标识一个 user, 比如 邮箱/手机号/userName

    用这个参数作为key存入redis, 对应的value为登陆错误的次数, string 类型, 并设置过期时间为 1min. 当获取到的 value == "4" , 说明当前请求为第 5 次登陆异常, 锁定.

    所谓的锁定, 就是将对应的value设置为某个标识符, 比如"lock", 并设置过期时间为 1h

    核心代码

    定义一个注解, 用来标识需要登陆次数校验的方法

    package io.github.xiaoyureed.redispractice.anno;
    import java.lang.annotation.*;
    @Documented
    @Target({ElementType.METHOD})
    @Retention(RetentionPolicy.RUNTIME)
    public @interface RedisLimit {
      /**
       * 标识参数名, 必须是请求参数中的一个
       */
      String identifier();
      /**
       * 在多长时间内监控, 如希望在 60s 内尝试
       * 次数限制为5次, 那么 watch=60; unit: s
       */
      long watch();
      /**
       * 锁定时长, unit: s
       */
      long lock();
      /**
       * 错误的尝试次数
       */
      int times();
    }

    编写切面, 在目标方法前后进行校验, 处理...

    package io.github.xiaoyureed.redispractice.aop;
    @Component
    @Aspect
    // Ensure that current advice is outer compared with ControllerAOP
    // so we can handling login limitation Exception in this aop advice.
    //@Order(9)
    @Slf4j
    public class RedisLimitAOP {
      @Autowired
      private StringRedisTemplate stringRedisTemplate;
      @Around("@annotation(io.github.xiaoyureed.redispractice.anno.RedisLimit)")
      public Object handleLimit(ProceedingJoinPoint joinPoint) {
        MethodSignature methodSignature = (MethodSignature) joinPoint.getSignature();
        final Method   method     = methodSignature.getMethod();
        final RedisLimit redisLimitAnno = method.getAnnotation(RedisLimit.class);// 貌似可以直接在方法参数中注入 todo
        final String identifier = redisLimitAnno.identifier();
        final long  watch   = redisLimitAnno.watch();
        final int  times   = redisLimitAnno.times();
        final long  lock    = redisLimitAnno.lock();
        // final ServletRequestAttributes att       = (ServletRequestAttributes) RequestContextHolder.currentRequestAttributes();
        // final HttpServletRequest    request     = att.getRequest();
        // final String          identifierValue = request.getParameter(identifier);
        String identifierValue = null;
        try {
          final Object arg      = joinPoint.getArgs()[0];
          final Field declaredField = arg.getClass().getDeclaredField(identifier);
          declaredField.setAccessible(true);
          identifierValue = (String) declaredField.get(arg);
        } catch (NoSuchFieldException e) {
          log.error(">>> invalid identifier [{}], cannot find this field in request params", identifier);
        } catch (IllegalAccessException e) {
          e.printStackTrace();
        }
        if (StringUtils.isBlank(identifierValue)) {
          log.error(">>> the value of RedisLimit.identifier cannot be blank, invalid identifier: {}", identifier);
        }
        // check User locked
        final ValueOperationsString, String> ssOps = stringRedisTemplate.opsForValue();
        final String             flag = ssOps.get(identifierValue);
        if (flag != null  "lock".contentEquals(flag)) {
          final BaseResp result = new BaseResp();
          result.setErrMsg("user locked");
          result.setCode("1");
          return new ResponseEntity>(result, HttpStatus.OK);
        }
        ResponseEntity result;
        try {
          result = (ResponseEntity) joinPoint.proceed();
        } catch (Throwable e) {
          result = handleLoginException(e, identifierValue, watch, times, lock);
        }
        return result;
      }
      private ResponseEntity handleLoginException(Throwable e, String identifierValue, long watch, int times, long lock) {
        final BaseResp result = new BaseResp();
        result.setCode("1");
        if (e instanceof LoginException) {
          log.info(">>> handle login exception...");
          final ValueOperationsString, String> ssOps = stringRedisTemplate.opsForValue();
          Boolean                exist = stringRedisTemplate.hasKey(identifierValue);
          // key doesn't exist, so it is the first login failure
          if (exist == null || !exist) {
            ssOps.set(identifierValue, "1", watch, TimeUnit.SECONDS);
            result.setErrMsg(e.getMessage());
            return new ResponseEntity>(result, HttpStatus.OK);
          }
          String count = ssOps.get(identifierValue);
          // has been reached the limitation
          if (Integer.parseInt(count) + 1 == times) {
            log.info(">>> [{}] has been reached the limitation and will be locked for {}s", identifierValue, lock);
            ssOps.set(identifierValue, "lock", lock, TimeUnit.SECONDS);
            result.setErrMsg("user locked");
            return new ResponseEntity>(result, HttpStatus.OK);
          }
          ssOps.increment(identifierValue);
          result.setErrMsg(e.getMessage() + "; you have try " + ssOps.get(identifierValue) + "times.");
        }
        log.error(">>> RedisLimitAOP cannot handle {}", e.getClass().getName());
        return new ResponseEntity>(result, HttpStatus.OK);
      }
    }

    这样使用:

    package io.github.xiaoyureed.redispractice.web;
    @RestController
    public class SessionResources {
      @Autowired
      private SessionService sessionService;
      /**
       * 1 min 之内尝试超过5次, 锁定 user 1h
       */
      @RedisLimit(identifier = "name", watch = 30, times = 5, lock = 10)
      @RequestMapping(value = "/session", method = RequestMethod.POST)
      public ResponseEntityLoginResp> login(@Validated @RequestBody LoginReq req) {
        return new ResponseEntity>(sessionService.login(req), HttpStatus.OK);
      }
    }

    references

    https://github.com/xiaoyureed/redis-login-limitation

    总结

    以上所述是小编给大家介绍的redis 实现登陆次数限制的思路详解,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对脚本之家网站的支持!
    如果你觉得本文对你有帮助,欢迎转载,烦请注明出处,谢谢!

    您可能感兴趣的文章:
    • PHP+Redis 消息队列 实现高并发下注册人数统计的实例
    • 利用Redis统计网站在线活跃用户的方法
    • PHP使用redis实现统计缓存mysql压力的方法
    • Redis中统计各种数据大小的方法
    • 基于redis实现token验证用户是否登陆
    • redis开启和禁用登陆密码校验的方法
    • 基于Redis位图实现系统用户登录统计
    上一篇:Linux下redis的安装与使用图文教程
    下一篇:基于Redis实现每日登录失败次数限制
  • 相关文章
  • 

    © 2016-2020 巨人网络通讯 版权所有

    《增值电信业务经营许可证》 苏ICP备15040257号-8

    redis 实现登陆次数限制的思路详解 redis,实现,登陆,次数,限制,