parent
03cf98d3c9
commit
5c155f5f11
@ -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