8 changed files with 225 additions and 15 deletions
@ -0,0 +1,40 @@ |
|||
package com.ruoyi.common.annotation; |
|||
|
|||
import java.lang.annotation.Documented; |
|||
import java.lang.annotation.ElementType; |
|||
import java.lang.annotation.Retention; |
|||
import java.lang.annotation.RetentionPolicy; |
|||
import java.lang.annotation.Target; |
|||
import com.ruoyi.common.constant.Constants; |
|||
import com.ruoyi.common.enums.LimitType; |
|||
|
|||
/** |
|||
* 限流注解 |
|||
* |
|||
* @author ruoyi |
|||
*/ |
|||
@Target(ElementType.METHOD) |
|||
@Retention(RetentionPolicy.RUNTIME) |
|||
@Documented |
|||
public @interface RateLimiter |
|||
{ |
|||
/** |
|||
* 限流key |
|||
*/ |
|||
public String key() default Constants.RATE_LIMIT_KEY; |
|||
|
|||
/** |
|||
* 限流时间,单位秒 |
|||
*/ |
|||
public int time() default 60; |
|||
|
|||
/** |
|||
* 限流次数 |
|||
*/ |
|||
public int count() default 100; |
|||
|
|||
/** |
|||
* 限流类型 |
|||
*/ |
|||
public LimitType limitType() default LimitType.DEFAULT; |
|||
} |
@ -0,0 +1,20 @@ |
|||
package com.ruoyi.common.enums; |
|||
|
|||
/** |
|||
* 限流类型 |
|||
* |
|||
* @author ruoyi |
|||
*/ |
|||
|
|||
public enum LimitType |
|||
{ |
|||
/** |
|||
* 默认策略全局限流 |
|||
*/ |
|||
DEFAULT, |
|||
|
|||
/** |
|||
* 根据请求者IP进行限流 |
|||
*/ |
|||
IP |
|||
} |
@ -0,0 +1,116 @@ |
|||
package com.ruoyi.framework.aspectj; |
|||
|
|||
import java.lang.reflect.Method; |
|||
import java.util.Collections; |
|||
import java.util.List; |
|||
import org.aspectj.lang.JoinPoint; |
|||
import org.aspectj.lang.Signature; |
|||
import org.aspectj.lang.annotation.Aspect; |
|||
import org.aspectj.lang.annotation.Before; |
|||
import org.aspectj.lang.annotation.Pointcut; |
|||
import org.aspectj.lang.reflect.MethodSignature; |
|||
import org.slf4j.Logger; |
|||
import org.slf4j.LoggerFactory; |
|||
import org.springframework.beans.factory.annotation.Autowired; |
|||
import org.springframework.data.redis.core.RedisTemplate; |
|||
import org.springframework.data.redis.core.script.RedisScript; |
|||
import org.springframework.stereotype.Component; |
|||
import com.ruoyi.common.annotation.RateLimiter; |
|||
import com.ruoyi.common.enums.LimitType; |
|||
import com.ruoyi.common.exception.ServiceException; |
|||
import com.ruoyi.common.utils.ServletUtils; |
|||
import com.ruoyi.common.utils.StringUtils; |
|||
import com.ruoyi.common.utils.ip.IpUtils; |
|||
|
|||
/** |
|||
* 限流处理 |
|||
* |
|||
* @author ruoyi |
|||
*/ |
|||
@Aspect |
|||
@Component |
|||
public class RateLimiterAspect |
|||
{ |
|||
private static final Logger log = LoggerFactory.getLogger(RateLimiterAspect.class); |
|||
|
|||
private RedisTemplate<Object, Object> redisTemplate; |
|||
|
|||
private RedisScript<Long> limitScript; |
|||
|
|||
@Autowired |
|||
public void setRedisTemplate1(RedisTemplate<Object, Object> redisTemplate) |
|||
{ |
|||
this.redisTemplate = redisTemplate; |
|||
} |
|||
|
|||
@Autowired |
|||
public void setLimitScript(RedisScript<Long> limitScript) |
|||
{ |
|||
this.limitScript = limitScript; |
|||
} |
|||
|
|||
// 配置织入点
|
|||
@Pointcut("@annotation(com.ruoyi.common.annotation.RateLimiter)") |
|||
public void rateLimiterPointCut() |
|||
{ |
|||
} |
|||
|
|||
@Before("rateLimiterPointCut()") |
|||
public void doBefore(JoinPoint point) throws Throwable |
|||
{ |
|||
RateLimiter rateLimiter = getAnnotationRateLimiter(point); |
|||
String key = rateLimiter.key(); |
|||
int time = rateLimiter.time(); |
|||
int count = rateLimiter.count(); |
|||
|
|||
String combineKey = getCombineKey(rateLimiter, point); |
|||
List<Object> keys = Collections.singletonList(combineKey); |
|||
try |
|||
{ |
|||
Long number = redisTemplate.execute(limitScript, keys, count, time); |
|||
if (StringUtils.isNull(number) || number.intValue() > count) |
|||
{ |
|||
throw new ServiceException("访问过于频繁,请稍后再试"); |
|||
} |
|||
log.info("限制请求'{}',当前请求'{}',缓存key'{}'", count, number.intValue(), key); |
|||
} |
|||
catch (ServiceException e) |
|||
{ |
|||
throw e; |
|||
} |
|||
catch (Exception e) |
|||
{ |
|||
throw new RuntimeException("服务器限流异常,请稍后再试"); |
|||
} |
|||
} |
|||
|
|||
/** |
|||
* 是否存在注解,如果存在就获取 |
|||
*/ |
|||
private RateLimiter getAnnotationRateLimiter(JoinPoint joinPoint) |
|||
{ |
|||
Signature signature = joinPoint.getSignature(); |
|||
MethodSignature methodSignature = (MethodSignature) signature; |
|||
Method method = methodSignature.getMethod(); |
|||
|
|||
if (method != null) |
|||
{ |
|||
return method.getAnnotation(RateLimiter.class); |
|||
} |
|||
return null; |
|||
} |
|||
|
|||
public String getCombineKey(RateLimiter rateLimiter, JoinPoint point) |
|||
{ |
|||
StringBuffer stringBuffer = new StringBuffer(rateLimiter.key()); |
|||
if (rateLimiter.limitType() == LimitType.IP) |
|||
{ |
|||
stringBuffer.append(IpUtils.getIpAddr(ServletUtils.getRequest())); |
|||
} |
|||
MethodSignature signature = (MethodSignature) point.getSignature(); |
|||
Method method = signature.getMethod(); |
|||
Class<?> targetClass = method.getDeclaringClass(); |
|||
stringBuffer.append("-").append(targetClass.getName()).append("- ").append(method.getName()); |
|||
return stringBuffer.toString(); |
|||
} |
|||
} |
Loading…
Reference in new issue