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