mirror of
				https://github.com/continew-org/continew-starter.git
				synced 2025-10-25 08:57:12 +08:00 
			
		
		
		
	refactor(security/limiter): 重构限流器,支持自定义限流器名称生成器
This commit is contained in:
		| @@ -28,7 +28,7 @@ import org.springframework.boot.context.properties.ConfigurationProperties; | ||||
|  * @author Charles7c | ||||
|  * @since 1.0.0 | ||||
|  */ | ||||
| @ConfigurationProperties(prefix = "spring.data.redisson") | ||||
| @ConfigurationProperties("spring.data.redisson") | ||||
| public class RedissonProperties { | ||||
|  | ||||
|     /** | ||||
|   | ||||
| @@ -262,6 +262,16 @@ public class StringConstants { | ||||
|      */ | ||||
|     public static final String CHINESE_COMMA = ","; | ||||
|  | ||||
|     /** | ||||
|      * 圆括号(左) {@code "("} | ||||
|      */ | ||||
|     public static final String ROUND_BRACKET_START = "("; | ||||
|  | ||||
|     /** | ||||
|      * 圆括号(右) {@code ")"} | ||||
|      */ | ||||
|     public static final String ROUND_BRACKET_END = ")"; | ||||
|  | ||||
|     /** | ||||
|      * 路径模式 | ||||
|      */ | ||||
|   | ||||
| @@ -8,32 +8,27 @@ | ||||
|     </parent> | ||||
|  | ||||
|     <artifactId>continew-starter-security-limiter</artifactId> | ||||
|     <description>ContiNew Starter 安全模块 - 限流模块</description> | ||||
|     <description>ContiNew Starter 安全模块 - 限流</description> | ||||
|  | ||||
|     <dependencies> | ||||
|         <!-- Redisson(不仅仅是一个 Redis Java 客户端,Redisson 充分的利用了 Redis 键值数据库提供的一系列优势,为使用者提供了一系列具有分布式特性的常用工具:分布式锁、限流器等) --> | ||||
|         <dependency> | ||||
|             <groupId>org.redisson</groupId> | ||||
|             <artifactId>redisson-spring-boot-starter</artifactId> | ||||
|         </dependency> | ||||
|         <!--aop--> | ||||
|         <dependency> | ||||
|             <groupId>org.aspectj</groupId> | ||||
|             <artifactId>aspectjrt</artifactId> | ||||
|             <version>1.9.4</version> | ||||
|         </dependency> | ||||
|         <dependency> | ||||
|             <groupId>org.aspectj</groupId> | ||||
|             <artifactId>aspectjweaver</artifactId> | ||||
|             <version>1.9.4</version> | ||||
|         </dependency> | ||||
|         <!--hutool--> | ||||
|         <dependency> | ||||
|             <groupId>cn.hutool</groupId> | ||||
|             <artifactId>hutool-all</artifactId> | ||||
|             <groupId>org.springframework.boot</groupId> | ||||
|             <artifactId>spring-boot-starter-aop</artifactId> | ||||
|         </dependency> | ||||
|  | ||||
|         <!--Redisson缓存 模块--> | ||||
|         <!-- Hutool 加密解密模块(封装 JDK 中加密解密算法) --> | ||||
|         <dependency> | ||||
|             <groupId>cn.hutool</groupId> | ||||
|             <artifactId>hutool-crypto</artifactId> | ||||
|         </dependency> | ||||
|  | ||||
|         <!-- Web 模块 --> | ||||
|         <dependency> | ||||
|             <groupId>top.continew</groupId> | ||||
|             <artifactId>continew-starter-web</artifactId> | ||||
|         </dependency> | ||||
|  | ||||
|         <!-- 缓存模块 - Redisson --> | ||||
|         <dependency> | ||||
|             <groupId>top.continew</groupId> | ||||
|             <artifactId>continew-starter-cache-redisson</artifactId> | ||||
|   | ||||
| @@ -33,40 +33,37 @@ import java.lang.annotation.*; | ||||
| public @interface RateLimiter { | ||||
|  | ||||
|     /** | ||||
|      * LimitType 限流模式 | ||||
|      * DEFAULT 全局限流 | ||||
|      * IP IP限流 | ||||
|      * CLUSTER 实例限流 | ||||
|      * 类型 | ||||
|      */ | ||||
|     LimitType limitType() default LimitType.DEFAULT; | ||||
|     LimitType type() default LimitType.DEFAULT; | ||||
|  | ||||
|     /** | ||||
|      * 缓存实例名称 | ||||
|      * 名称 | ||||
|      */ | ||||
|     String name() default ""; | ||||
|  | ||||
|     /** | ||||
|      * 限流key 支持 Spring EL 表达式 | ||||
|      * 键(支持 Spring EL 表达式) | ||||
|      */ | ||||
|     String key() default ""; | ||||
|  | ||||
|     /** | ||||
|      * 单位时间产生的令牌数 | ||||
|      * 速率(指定时间间隔产生的令牌数) | ||||
|      */ | ||||
|     int rate() default Integer.MAX_VALUE; | ||||
|  | ||||
|     /** | ||||
|      * 限流时间 | ||||
|      * 速率间隔(时间间隔) | ||||
|      */ | ||||
|     int rateInterval() default 0; | ||||
|     int interval() default 0; | ||||
|  | ||||
|     /** | ||||
|      * 时间单位,默认毫秒 | ||||
|      * 速率间隔时间单位(默认:毫秒) | ||||
|      */ | ||||
|     RateIntervalUnit timeUnit() default RateIntervalUnit.MILLISECONDS; | ||||
|     RateIntervalUnit unit() default RateIntervalUnit.MILLISECONDS; | ||||
|  | ||||
|     /** | ||||
|      * 拒绝请求时的提示信息 | ||||
|      * 提示信息 | ||||
|      */ | ||||
|     String message() default "您操作过于频繁,请稍后再试!"; | ||||
|     String message() default "操作过于频繁,请稍后再试"; | ||||
| } | ||||
| @@ -19,7 +19,7 @@ package top.continew.starter.security.limiter.annotation; | ||||
| import java.lang.annotation.*; | ||||
|  | ||||
| /** | ||||
|  * 限流组 | ||||
|  * 限流组注解 | ||||
|  *  | ||||
|  * @author KAI | ||||
|  * @since 2.2.0 | ||||
| @@ -28,8 +28,9 @@ import java.lang.annotation.*; | ||||
| @Retention(RetentionPolicy.RUNTIME) | ||||
| @Target(ElementType.METHOD) | ||||
| public @interface RateLimiters { | ||||
|  | ||||
|     /** | ||||
|      * 用于管理多个 RateLimiter | ||||
|      * 限流组 | ||||
|      */ | ||||
|     RateLimiter[] value(); | ||||
| } | ||||
|   | ||||
| @@ -1,265 +0,0 @@ | ||||
| /* | ||||
|  * Copyright (c) 2022-present Charles7c Authors. All Rights Reserved. | ||||
|  * <p> | ||||
|  * Licensed under the GNU LESSER GENERAL PUBLIC LICENSE 3.0; | ||||
|  * you may not use this file except in compliance with the License. | ||||
|  * You may obtain a copy of the License at | ||||
|  * <p> | ||||
|  * http://www.gnu.org/licenses/lgpl.html | ||||
|  * <p> | ||||
|  * Unless required by applicable law or agreed to in writing, software | ||||
|  * distributed under the License is distributed on an "AS IS" BASIS, | ||||
|  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
|  * See the License for the specific language governing permissions and | ||||
|  * limitations under the License. | ||||
|  */ | ||||
|  | ||||
| package top.continew.starter.security.limiter.aop; | ||||
|  | ||||
| import cn.hutool.crypto.SecureUtil; | ||||
| import cn.hutool.extra.servlet.JakartaServletUtil; | ||||
| import cn.hutool.extra.spring.SpringUtil; | ||||
| import org.aspectj.lang.JoinPoint; | ||||
| import org.aspectj.lang.ProceedingJoinPoint; | ||||
| import org.aspectj.lang.annotation.Around; | ||||
| import org.aspectj.lang.annotation.Aspect; | ||||
| import org.aspectj.lang.annotation.Pointcut; | ||||
| import org.aspectj.lang.reflect.MethodSignature; | ||||
| import org.redisson.api.*; | ||||
| import org.slf4j.Logger; | ||||
| import org.slf4j.LoggerFactory; | ||||
| import org.springframework.beans.factory.annotation.Autowired; | ||||
| import org.springframework.context.expression.BeanFactoryResolver; | ||||
| import org.springframework.context.expression.MethodBasedEvaluationContext; | ||||
| import org.springframework.core.DefaultParameterNameDiscoverer; | ||||
| import org.springframework.core.ParameterNameDiscoverer; | ||||
| import org.springframework.expression.Expression; | ||||
| import org.springframework.expression.ExpressionParser; | ||||
| import org.springframework.expression.ParserContext; | ||||
| import org.springframework.expression.common.TemplateParserContext; | ||||
| import org.springframework.expression.spel.standard.SpelExpressionParser; | ||||
| import org.springframework.stereotype.Component; | ||||
| import org.springframework.util.StringUtils; | ||||
| import top.continew.starter.security.limiter.annotation.RateLimiter; | ||||
| import top.continew.starter.security.limiter.annotation.RateLimiters; | ||||
| import top.continew.starter.security.limiter.autoconfigure.RateLimiterProperties; | ||||
| import top.continew.starter.security.limiter.enums.LimitType; | ||||
| import top.continew.starter.security.limiter.exception.RateLimiterException; | ||||
| import top.continew.starter.web.util.ServletUtils; | ||||
|  | ||||
| import java.util.Objects; | ||||
| import java.util.Optional; | ||||
| import java.util.concurrent.ConcurrentHashMap; | ||||
|  | ||||
| /** | ||||
|  * 限流 AOP 拦截器 | ||||
|  * | ||||
|  * @author KAI | ||||
|  * @since 2.2.0 | ||||
|  */ | ||||
| @Aspect | ||||
| @Component | ||||
| public class RateLimiterAspect { | ||||
|  | ||||
|     private static final Logger log = LoggerFactory.getLogger(RateLimiterAspect.class); | ||||
|  | ||||
|     private static final ConcurrentHashMap<String, RRateLimiter> rateLimiterCache = new ConcurrentHashMap<>(); | ||||
|  | ||||
|     private static final RedissonClient CLIENT = SpringUtil.getBean(RedissonClient.class); | ||||
|  | ||||
|     private final ParameterNameDiscoverer discoverer = new DefaultParameterNameDiscoverer(); | ||||
|     private final ExpressionParser parser = new SpelExpressionParser(); | ||||
|     private final ParserContext parserContext = new TemplateParserContext(); | ||||
|  | ||||
|     @Autowired | ||||
|     private RateLimiterProperties rateLimiterProperties; | ||||
|  | ||||
|     /** | ||||
|      * 单个限流注解切点 | ||||
|      */ | ||||
|     @Pointcut("@annotation(top.continew.starter.security.limiter.annotation.RateLimiter)") | ||||
|     public void rateLimiterSinglePointCut() { | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 多个限流注解切点 | ||||
|      */ | ||||
|     @Pointcut("@annotation(top.continew.starter.security.limiter.annotation.RateLimiters)") | ||||
|     public void rateLimiterBatchPointCut() { | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 环绕通知,处理单个限流注解 | ||||
|      * | ||||
|      * @param joinPoint   切点 | ||||
|      * @param rateLimiter 限流注解 | ||||
|      * @return 返回目标方法的执行结果 | ||||
|      * @throws Throwable 异常 | ||||
|      */ | ||||
|     @Around("@annotation(rateLimiter)") | ||||
|     public Object aroundSingle(ProceedingJoinPoint joinPoint, RateLimiter rateLimiter) throws Throwable { | ||||
|         // 未开启限流功能,直接执行目标方法 | ||||
|         if (!rateLimiterProperties.isEnabled()) { | ||||
|             return joinPoint.proceed(); | ||||
|         } | ||||
|         if (isRateLimited(joinPoint, rateLimiter)) { | ||||
|             throw new RateLimiterException(rateLimiter.message()); | ||||
|         } | ||||
|         return joinPoint.proceed(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 环绕通知,处理多个限流注解 | ||||
|      * | ||||
|      * @param joinPoint    切点 | ||||
|      * @param rateLimiters 多个限流注解 | ||||
|      * @return 返回目标方法的执行结果 | ||||
|      * @throws Throwable 异常 | ||||
|      */ | ||||
|     @Around("@annotation(rateLimiters)") | ||||
|     public Object aroundBatch(ProceedingJoinPoint joinPoint, RateLimiters rateLimiters) throws Throwable { | ||||
|         // 未开启限流功能,直接执行目标方法 | ||||
|         if (!rateLimiterProperties.isEnabled()) { | ||||
|             return joinPoint.proceed(); | ||||
|         } | ||||
|         for (RateLimiter rateLimiter : rateLimiters.value()) { | ||||
|             if (isRateLimited(joinPoint, rateLimiter)) { | ||||
|                 throw new RateLimiterException(rateLimiter.message()); | ||||
|             } | ||||
|         } | ||||
|         return joinPoint.proceed(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 执行限流逻辑 | ||||
|      * | ||||
|      * @param joinPoint   切点 | ||||
|      * @param rateLimiter 限流注解 | ||||
|      * @throws Throwable 异常 | ||||
|      */ | ||||
|     private boolean isRateLimited(ProceedingJoinPoint joinPoint, RateLimiter rateLimiter) throws Throwable { | ||||
|         try { | ||||
|             // 生成限流 Key | ||||
|             String redisKey = generateRedisKey(rateLimiter, joinPoint); | ||||
|             String encipherKey = SecureUtil.md5(redisKey); | ||||
|             // 确定限流类型 | ||||
|             RateType rateType = rateLimiter.limitType() == LimitType.CLUSTER ? RateType.PER_CLIENT : RateType.OVERALL; | ||||
|  | ||||
|             // 获取redisson限流实例 | ||||
|             RRateLimiter rRateLimiter = getRateLimiter(encipherKey); | ||||
|             RateIntervalUnit rateIntervalUnit = rateLimiter.timeUnit(); | ||||
|             int rateInterval = rateLimiter.rateInterval(); | ||||
|             int rate = rateLimiter.rate(); | ||||
|             // 判断是否需要更新限流器 | ||||
|             if (shouldUpdateRateLimiter(rRateLimiter, rateType, rate, rateInterval, rateIntervalUnit)) { | ||||
|                 // 更新限流器 | ||||
|                 rRateLimiter.setRate(rateType, rate, rateInterval, rateIntervalUnit); | ||||
|             } | ||||
|             // 尝试获取令牌 | ||||
|             return !rRateLimiter.tryAcquire(); | ||||
|         } catch (RateLimiterException e) { | ||||
|             throw e; | ||||
|         } catch (Exception e) { | ||||
|             throw new RuntimeException("服务器限流异常,请稍候再试", e); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 获取 Redisson RateLimiter 实例 | ||||
|      * | ||||
|      * @param key 限流器的 Key | ||||
|      * @return RateLimiter 实例 | ||||
|      */ | ||||
|     private RRateLimiter getRateLimiter(String key) { | ||||
|         RRateLimiter rRateLimiter = rateLimiterCache.get(key); | ||||
|         if (rRateLimiter == null) { | ||||
|             // 直接创建 RateLimiter 实例 | ||||
|             rRateLimiter = CLIENT.getRateLimiter(key); | ||||
|             rateLimiterCache.put(key, rRateLimiter); | ||||
|         } | ||||
|         return rRateLimiter; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 判断是否需要更新限流器配置 | ||||
|      * | ||||
|      * @param rRateLimiter     现有的限流器 | ||||
|      * @param rateType         限流类型(OVERALL:全局限流;PER_CLIENT:单机限流) | ||||
|      * @param rate             速率(指定时间间隔产生的令牌数) | ||||
|      * @param rateInterval     速率间隔 | ||||
|      * @param rateIntervalUnit 时间单位 | ||||
|      * @return 是否需要更新配置 | ||||
|      */ | ||||
|     private boolean shouldUpdateRateLimiter(RRateLimiter rRateLimiter, | ||||
|                                             RateType rateType, | ||||
|                                             long rate, | ||||
|                                             long rateInterval, | ||||
|                                             RateIntervalUnit rateIntervalUnit) { | ||||
|  | ||||
|         RateLimiterConfig config = rRateLimiter.getConfig(); | ||||
|         return !Objects.equals(config.getRateType(), rateType) || !Objects.equals(config.getRate(), rate) || !Objects | ||||
|             .equals(config.getRateInterval(), rateIntervalUnit.toMillis(rateInterval)); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 获取限流Key | ||||
|      * | ||||
|      * @param rateLimiter RateLimiter实例 | ||||
|      * @param point       切点 | ||||
|      * @return 限流Key | ||||
|      */ | ||||
|     private String generateRedisKey(RateLimiter rateLimiter, JoinPoint point) { | ||||
|         // 获取限流器配置的 key | ||||
|         String key = rateLimiter.key(); | ||||
|         // 如果 key 不为空,则解析表达式并获取最终的 key | ||||
|         key = Optional.ofNullable(key).map(k -> { | ||||
|             // 获取方法签名 | ||||
|             MethodSignature signature = (MethodSignature)point.getSignature(); | ||||
|             // 获取方法参数 | ||||
|             Object[] args = point.getArgs(); | ||||
|             // 创建表达式上下文 | ||||
|             MethodBasedEvaluationContext context = new MethodBasedEvaluationContext(null, signature | ||||
|                 .getMethod(), args, discoverer); | ||||
|             // 设置 Bean 解析器 | ||||
|             context.setBeanResolver(new BeanFactoryResolver(SpringUtil.getBeanFactory())); | ||||
|             // 解析表达式 | ||||
|             Expression expression; | ||||
|             if (StringUtils.startsWithIgnoreCase(k, parserContext.getExpressionPrefix()) && StringUtils | ||||
|                 .endsWithIgnoreCase(k, parserContext.getExpressionSuffix())) { | ||||
|                 expression = parser.parseExpression(k, parserContext); | ||||
|             } else { | ||||
|                 expression = parser.parseExpression(k); | ||||
|             } | ||||
|             // 获取表达式结果 | ||||
|             return expression.getValue(context, String.class); | ||||
|         }).orElse(key); | ||||
|  | ||||
|         // 拼接最终的 key | ||||
|         StringBuilder redisKey = new StringBuilder(rateLimiterProperties.getLimiterKey()).append(ServletUtils | ||||
|             .getRequest() | ||||
|             .getRequestURI()).append(":"); | ||||
|         //如果缓存name 不为空 则拼接上去 | ||||
|         String name = rateLimiter.name(); | ||||
|         if (StringUtils.hasText(name)) { | ||||
|             redisKey.append(name); | ||||
|             if (!name.endsWith(":")) { | ||||
|                 redisKey.append(":"); | ||||
|             } | ||||
|         } | ||||
|         // 根据限流类型添加不同的信息 | ||||
|         switch (rateLimiter.limitType()) { | ||||
|             case IP: | ||||
|                 // 获取请求 IP | ||||
|                 redisKey.append(JakartaServletUtil.getClientIP(ServletUtils.getRequest())).append(":"); | ||||
|                 break; | ||||
|             case CLUSTER: | ||||
|                 // 获取客户端实例 ID | ||||
|                 redisKey.append(CLIENT.getId()).append(":"); | ||||
|                 break; | ||||
|             default: | ||||
|                 break; | ||||
|         } | ||||
|         // 添加解析后的 key | ||||
|         return redisKey.append(key).toString(); | ||||
|     } | ||||
| } | ||||
| @@ -16,29 +16,45 @@ | ||||
|  | ||||
| package top.continew.starter.security.limiter.autoconfigure; | ||||
|  | ||||
| import jakarta.annotation.PostConstruct; | ||||
| import org.slf4j.Logger; | ||||
| import org.slf4j.LoggerFactory; | ||||
| import org.springframework.boot.autoconfigure.AutoConfiguration; | ||||
| import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; | ||||
| import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; | ||||
| import org.springframework.boot.context.properties.EnableConfigurationProperties; | ||||
| import org.springframework.context.annotation.Bean; | ||||
| import top.continew.starter.security.limiter.aop.RateLimiterAspect; | ||||
| import org.springframework.context.annotation.ComponentScan; | ||||
| import top.continew.starter.core.constant.PropertiesConstants; | ||||
| import top.continew.starter.security.limiter.core.DefaultRateLimiterNameGenerator; | ||||
| import top.continew.starter.security.limiter.core.RateLimiterNameGenerator; | ||||
|  | ||||
| /** | ||||
|  * 限流配置注入器 | ||||
|  * 限流器自动配置 | ||||
|  *  | ||||
|  * @author KAI | ||||
|  * @author Charles7c | ||||
|  * @since 2.2.0 | ||||
|  */ | ||||
|  | ||||
| @AutoConfiguration | ||||
| @EnableConfigurationProperties(RateLimiterProperties.class) | ||||
| @ComponentScan({"top.continew.starter.security.limiter.core"}) | ||||
| @ConditionalOnProperty(prefix = PropertiesConstants.SECURITY_LIMITER, name = PropertiesConstants.ENABLED, matchIfMissing = true) | ||||
| public class RateLimiterAutoConfiguration { | ||||
|  | ||||
|     private static final Logger log = LoggerFactory.getLogger(RateLimiterAutoConfiguration.class); | ||||
|  | ||||
|     /** | ||||
|      * 限流器名称生成器 | ||||
|      */ | ||||
|     @Bean | ||||
|     public RateLimiterAspect rateLimiterAspect() { | ||||
|         log.info("[ContiNew Starter] - Auto Configuration 'RateLimiterAspect' completed initialization."); | ||||
|         return new RateLimiterAspect(); | ||||
|     @ConditionalOnMissingBean | ||||
|     public RateLimiterNameGenerator nameGenerator() { | ||||
|         return new DefaultRateLimiterNameGenerator(); | ||||
|     } | ||||
|  | ||||
|     @PostConstruct | ||||
|     public void postConstruct() { | ||||
|         log.debug("[ContiNew Starter] - Auto Configuration 'Security-RateLimiter' completed initialization."); | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -17,7 +17,7 @@ | ||||
| package top.continew.starter.security.limiter.autoconfigure; | ||||
|  | ||||
| import org.springframework.boot.context.properties.ConfigurationProperties; | ||||
| import org.springframework.util.StringUtils; | ||||
| import top.continew.starter.core.constant.PropertiesConstants; | ||||
|  | ||||
| /** | ||||
|  * 限流器配置属性 | ||||
| @@ -25,32 +25,19 @@ import org.springframework.util.StringUtils; | ||||
|  * @author KAI | ||||
|  * @since 2.2.0 | ||||
|  */ | ||||
| @ConfigurationProperties(prefix = "continew-starter.security.limiter") | ||||
| @ConfigurationProperties(PropertiesConstants.SECURITY_LIMITER) | ||||
| public class RateLimiterProperties { | ||||
|     private boolean enabled = false; | ||||
|  | ||||
|     private String limiterKey = "RateLimiter:"; | ||||
|     /** | ||||
|      * Key 前缀 | ||||
|      */ | ||||
|     private String keyPrefix = "RateLimiter"; | ||||
|  | ||||
|     public boolean isEnabled() { | ||||
|         return enabled; | ||||
|     public String getKeyPrefix() { | ||||
|         return keyPrefix; | ||||
|     } | ||||
|  | ||||
|     public void setEnabled(boolean enabled) { | ||||
|         this.enabled = enabled; | ||||
|     } | ||||
|  | ||||
|     public String getLimiterKey() { | ||||
|         return limiterKey; | ||||
|     } | ||||
|  | ||||
|     public void setLimiterKey(String limiterKey) { | ||||
|         //不为空且不以":"结尾,则添加":" | ||||
|         if (StringUtils.hasText(limiterKey)) { | ||||
|             if (!limiterKey.endsWith(":")) { | ||||
|                 limiterKey = limiterKey + ":"; | ||||
|     public void setKeyPrefix(String keyPrefix) { | ||||
|         this.keyPrefix = keyPrefix; | ||||
|     } | ||||
| } | ||||
|         this.limiterKey = limiterKey; | ||||
|     } | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -0,0 +1,109 @@ | ||||
| /* | ||||
|  * Copyright (c) 2022-present Charles7c Authors. All Rights Reserved. | ||||
|  * <p> | ||||
|  * Licensed under the GNU LESSER GENERAL PUBLIC LICENSE 3.0; | ||||
|  * you may not use this file except in compliance with the License. | ||||
|  * You may obtain a copy of the License at | ||||
|  * <p> | ||||
|  * http://www.gnu.org/licenses/lgpl.html | ||||
|  * <p> | ||||
|  * Unless required by applicable law or agreed to in writing, software | ||||
|  * distributed under the License is distributed on an "AS IS" BASIS, | ||||
|  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
|  * See the License for the specific language governing permissions and | ||||
|  * limitations under the License. | ||||
|  */ | ||||
|  | ||||
| package top.continew.starter.security.limiter.core; | ||||
|  | ||||
| import cn.hutool.core.util.ClassUtil; | ||||
| import top.continew.starter.core.constant.StringConstants; | ||||
|  | ||||
| import java.lang.reflect.Method; | ||||
| import java.util.concurrent.ConcurrentHashMap; | ||||
|  | ||||
| /** | ||||
|  * 默认限流器名称生成器 | ||||
|  * | ||||
|  * @author Charles7c | ||||
|  * @since 2.2.0 | ||||
|  */ | ||||
| public class DefaultRateLimiterNameGenerator implements RateLimiterNameGenerator { | ||||
|  | ||||
|     protected final ConcurrentHashMap<Method, String> nameMap = new ConcurrentHashMap<>(); | ||||
|  | ||||
|     @Override | ||||
|     public String generate(Object target, Method method, Object... args) { | ||||
|         return nameMap.computeIfAbsent(method, key -> { | ||||
|             final StringBuilder nameSb = new StringBuilder(); | ||||
|             String className = method.getDeclaringClass().getName(); | ||||
|             nameSb.append(ClassUtil.getShortClassName(className)); | ||||
|             nameSb.append(StringConstants.DOT); | ||||
|             nameSb.append(method.getName()); | ||||
|             nameSb.append(StringConstants.ROUND_BRACKET_START); | ||||
|             for (Class<?> clazz : method.getParameterTypes()) { | ||||
|                 this.getDescriptor(nameSb, clazz); | ||||
|             } | ||||
|             nameSb.append(StringConstants.ROUND_BRACKET_END); | ||||
|             String str = nameSb.toString(); | ||||
|             nameMap.put(method, str); | ||||
|             return str; | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 获取指定数据类型的描述 | ||||
|      * | ||||
|      * @param sb        名称字符串缓存 | ||||
|      * @param typeClass 数据类型 | ||||
|      */ | ||||
|     private void getDescriptor(final StringBuilder sb, final Class<?> typeClass) { | ||||
|         Class<?> clazz = typeClass; | ||||
|         while (true) { | ||||
|             if (clazz.isPrimitive()) { | ||||
|                 sb.append(this.getPrimitiveChar(clazz)); | ||||
|                 return; | ||||
|             } else if (clazz.isArray()) { | ||||
|                 sb.append(StringConstants.BRACKET_START); | ||||
|                 clazz = clazz.getComponentType(); | ||||
|             } else { | ||||
|                 sb.append('L'); | ||||
|                 String name = clazz.getName(); | ||||
|                 name = ClassUtil.getShortClassName(name); | ||||
|                 sb.append(name); | ||||
|                 sb.append(StringConstants.COLON); | ||||
|                 return; | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 根据基本数据获取类型字符 | ||||
|      * | ||||
|      * @param clazz 数据类型 | ||||
|      * @return 类型字符 | ||||
|      */ | ||||
|     private char getPrimitiveChar(Class<?> clazz) { | ||||
|         char c; | ||||
|         if (clazz == Integer.TYPE) { | ||||
|             c = 'I'; | ||||
|         } else if (clazz == Void.TYPE) { | ||||
|             c = 'V'; | ||||
|         } else if (clazz == Boolean.TYPE) { | ||||
|             c = 'Z'; | ||||
|         } else if (clazz == Byte.TYPE) { | ||||
|             c = 'B'; | ||||
|         } else if (clazz == Character.TYPE) { | ||||
|             c = 'C'; | ||||
|         } else if (clazz == Short.TYPE) { | ||||
|             c = 'S'; | ||||
|         } else if (clazz == Double.TYPE) { | ||||
|             c = 'D'; | ||||
|         } else if (clazz == Float.TYPE) { | ||||
|             c = 'F'; | ||||
|         } else { | ||||
|             c = 'J'; | ||||
|         } | ||||
|         return c; | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,200 @@ | ||||
| /* | ||||
|  * Copyright (c) 2022-present Charles7c Authors. All Rights Reserved. | ||||
|  * <p> | ||||
|  * Licensed under the GNU LESSER GENERAL PUBLIC LICENSE 3.0; | ||||
|  * you may not use this file except in compliance with the License. | ||||
|  * You may obtain a copy of the License at | ||||
|  * <p> | ||||
|  * http://www.gnu.org/licenses/lgpl.html | ||||
|  * <p> | ||||
|  * Unless required by applicable law or agreed to in writing, software | ||||
|  * distributed under the License is distributed on an "AS IS" BASIS, | ||||
|  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
|  * See the License for the specific language governing permissions and | ||||
|  * limitations under the License. | ||||
|  */ | ||||
|  | ||||
| package top.continew.starter.security.limiter.core; | ||||
|  | ||||
| import cn.hutool.core.convert.Convert; | ||||
| import cn.hutool.core.text.CharSequenceUtil; | ||||
| import cn.hutool.core.util.ObjectUtil; | ||||
| import cn.hutool.extra.servlet.JakartaServletUtil; | ||||
| import org.aspectj.lang.JoinPoint; | ||||
| import org.aspectj.lang.ProceedingJoinPoint; | ||||
| import org.aspectj.lang.annotation.Around; | ||||
| import org.aspectj.lang.annotation.Aspect; | ||||
| import org.aspectj.lang.annotation.Pointcut; | ||||
| import org.aspectj.lang.reflect.MethodSignature; | ||||
| import org.redisson.api.*; | ||||
| import org.springframework.stereotype.Component; | ||||
| import top.continew.starter.cache.redisson.util.RedisUtils; | ||||
| import top.continew.starter.core.constant.StringConstants; | ||||
| import top.continew.starter.core.util.expression.ExpressionUtils; | ||||
| import top.continew.starter.security.limiter.annotation.RateLimiter; | ||||
| import top.continew.starter.security.limiter.annotation.RateLimiters; | ||||
| import top.continew.starter.security.limiter.autoconfigure.RateLimiterProperties; | ||||
| import top.continew.starter.security.limiter.enums.LimitType; | ||||
| import top.continew.starter.security.limiter.exception.RateLimiterException; | ||||
| import top.continew.starter.web.util.ServletUtils; | ||||
|  | ||||
| import java.lang.reflect.Method; | ||||
| import java.util.Objects; | ||||
| import java.util.concurrent.ConcurrentHashMap; | ||||
|  | ||||
| /** | ||||
|  * 限流器切面 | ||||
|  * | ||||
|  * @author KAI | ||||
|  * @author Charles7c | ||||
|  * @since 2.2.0 | ||||
|  */ | ||||
| @Aspect | ||||
| @Component | ||||
| public class RateLimiterAspect { | ||||
|  | ||||
|     private static final ConcurrentHashMap<String, RRateLimiter> RATE_LIMITER_CACHE = new ConcurrentHashMap<>(); | ||||
|     private final RateLimiterProperties properties; | ||||
|     private final RateLimiterNameGenerator nameGenerator; | ||||
|     private final RedissonClient redissonClient; | ||||
|  | ||||
|     public RateLimiterAspect(RateLimiterProperties properties, | ||||
|                              RateLimiterNameGenerator nameGenerator, | ||||
|                              RedissonClient redissonClient) { | ||||
|         this.properties = properties; | ||||
|         this.nameGenerator = nameGenerator; | ||||
|         this.redissonClient = redissonClient; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 单个限流注解切点 | ||||
|      */ | ||||
|     @Pointcut("@annotation(top.continew.starter.security.limiter.annotation.RateLimiter)") | ||||
|     public void rateLimiterPointCut() { | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 多个限流注解切点 | ||||
|      */ | ||||
|     @Pointcut("@annotation(top.continew.starter.security.limiter.annotation.RateLimiters)") | ||||
|     public void rateLimitersPointCut() { | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 单限流场景 | ||||
|      * | ||||
|      * @param joinPoint   切点 | ||||
|      * @param rateLimiter 限流注解 | ||||
|      * @return 目标方法的执行结果 | ||||
|      * @throws Throwable / | ||||
|      */ | ||||
|     @Around("@annotation(rateLimiter)") | ||||
|     public Object aroundRateLimiter(ProceedingJoinPoint joinPoint, RateLimiter rateLimiter) throws Throwable { | ||||
|         if (isRateLimited(joinPoint, rateLimiter)) { | ||||
|             throw new RateLimiterException(rateLimiter.message()); | ||||
|         } | ||||
|         return joinPoint.proceed(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 多限流场景 | ||||
|      * | ||||
|      * @param joinPoint    切点 | ||||
|      * @param rateLimiters 限流组注解 | ||||
|      * @return 目标方法的执行结果 | ||||
|      * @throws Throwable / | ||||
|      */ | ||||
|     @Around("@annotation(rateLimiters)") | ||||
|     public Object aroundRateLimiters(ProceedingJoinPoint joinPoint, RateLimiters rateLimiters) throws Throwable { | ||||
|         for (RateLimiter rateLimiter : rateLimiters.value()) { | ||||
|             if (isRateLimited(joinPoint, rateLimiter)) { | ||||
|                 throw new RateLimiterException(rateLimiter.message()); | ||||
|             } | ||||
|         } | ||||
|         return joinPoint.proceed(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 是否需要限流 | ||||
|      * | ||||
|      * @param joinPoint   切点 | ||||
|      * @param rateLimiter 限流注解 | ||||
|      * @return true: 需要限流;false:不需要限流 | ||||
|      */ | ||||
|     private boolean isRateLimited(ProceedingJoinPoint joinPoint, RateLimiter rateLimiter) { | ||||
|         try { | ||||
|             String cacheKey = this.getCacheKey(joinPoint, rateLimiter); | ||||
|             RRateLimiter rRateLimiter = RATE_LIMITER_CACHE.computeIfAbsent(cacheKey, key -> redissonClient | ||||
|                 .getRateLimiter(cacheKey)); | ||||
|             // 限流器配置 | ||||
|             RateType rateType = rateLimiter.type() == LimitType.CLUSTER ? RateType.PER_CLIENT : RateType.OVERALL; | ||||
|             int rate = rateLimiter.rate(); | ||||
|             int rateInterval = rateLimiter.interval(); | ||||
|             RateIntervalUnit rateIntervalUnit = rateLimiter.unit(); | ||||
|             // 判断是否需要更新限流器 | ||||
|             if (this.isConfigurationUpdateNeeded(rRateLimiter, rateType, rate, rateInterval, rateIntervalUnit)) { | ||||
|                 // 更新限流器 | ||||
|                 rRateLimiter.setRate(rateType, rate, rateInterval, rateIntervalUnit); | ||||
|             } | ||||
|             // 尝试获取令牌 | ||||
|             return !rRateLimiter.tryAcquire(); | ||||
|         } catch (Exception e) { | ||||
|             throw new RateLimiterException("服务器限流异常,请稍候再试", e); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 获取限流缓存 Key | ||||
|      * | ||||
|      * @param joinPoint   切点 | ||||
|      * @param rateLimiter 限流注解 | ||||
|      * @return 限流缓存 Key | ||||
|      */ | ||||
|     private String getCacheKey(JoinPoint joinPoint, RateLimiter rateLimiter) { | ||||
|         Object target = joinPoint.getTarget(); | ||||
|         MethodSignature methodSignature = (MethodSignature)joinPoint.getSignature(); | ||||
|         Method method = methodSignature.getMethod(); | ||||
|         Object[] args = joinPoint.getArgs(); | ||||
|         // 获取限流名称 | ||||
|         String name = rateLimiter.name(); | ||||
|         if (CharSequenceUtil.isBlank(name)) { | ||||
|             name = nameGenerator.generate(target, method, args); | ||||
|         } | ||||
|         // 解析限流 Key | ||||
|         String key = rateLimiter.key(); | ||||
|         if (CharSequenceUtil.isNotBlank(key)) { | ||||
|             Object eval = ExpressionUtils.eval(key, target, method, args); | ||||
|             if (ObjectUtil.isNull(eval)) { | ||||
|                 throw new RateLimiterException("限流 Key 解析错误"); | ||||
|             } | ||||
|             key = Convert.toStr(eval); | ||||
|         } | ||||
|         // 获取后缀 | ||||
|         String suffix = switch (rateLimiter.type()) { | ||||
|             case IP -> JakartaServletUtil.getClientIP(ServletUtils.getRequest()); | ||||
|             case CLUSTER -> redissonClient.getId(); | ||||
|             default -> StringConstants.EMPTY; | ||||
|         }; | ||||
|         return RedisUtils.formatKey(properties.getKeyPrefix(), name, key, suffix); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 判断是否需要更新限流器配置 | ||||
|      * | ||||
|      * @param rRateLimiter     限流器 | ||||
|      * @param rateType         限流类型(OVERALL:全局限流;PER_CLIENT:单机限流) | ||||
|      * @param rate             速率(指定时间间隔产生的令牌数) | ||||
|      * @param rateInterval     速率间隔 | ||||
|      * @param rateIntervalUnit 时间单位 | ||||
|      * @return 是否需要更新配置 | ||||
|      */ | ||||
|     private boolean isConfigurationUpdateNeeded(RRateLimiter rRateLimiter, | ||||
|                                                 RateType rateType, | ||||
|                                                 long rate, | ||||
|                                                 long rateInterval, | ||||
|                                                 RateIntervalUnit rateIntervalUnit) { | ||||
|         RateLimiterConfig config = rRateLimiter.getConfig(); | ||||
|         return !Objects.equals(config.getRateType(), rateType) || !Objects.equals(config.getRate(), rate) || !Objects | ||||
|             .equals(config.getRateInterval(), rateIntervalUnit.toMillis(rateInterval)); | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,39 @@ | ||||
| /* | ||||
|  * Copyright (c) 2022-present Charles7c Authors. All Rights Reserved. | ||||
|  * <p> | ||||
|  * Licensed under the GNU LESSER GENERAL PUBLIC LICENSE 3.0; | ||||
|  * you may not use this file except in compliance with the License. | ||||
|  * You may obtain a copy of the License at | ||||
|  * <p> | ||||
|  * http://www.gnu.org/licenses/lgpl.html | ||||
|  * <p> | ||||
|  * Unless required by applicable law or agreed to in writing, software | ||||
|  * distributed under the License is distributed on an "AS IS" BASIS, | ||||
|  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
|  * See the License for the specific language governing permissions and | ||||
|  * limitations under the License. | ||||
|  */ | ||||
|  | ||||
| package top.continew.starter.security.limiter.core; | ||||
|  | ||||
| import java.lang.reflect.Method; | ||||
|  | ||||
| /** | ||||
|  * 限流器名称生成器 | ||||
|  * | ||||
|  * @author Charles7c | ||||
|  * @since 2.2.0 | ||||
|  */ | ||||
| @FunctionalInterface | ||||
| public interface RateLimiterNameGenerator { | ||||
|  | ||||
|     /** | ||||
|      * Generate a rate limiter name for the given method and its parameters. | ||||
|      * | ||||
|      * @param target the target instance | ||||
|      * @param method the method being called | ||||
|      * @param args   the method parameters (with any var-args expanded) | ||||
|      * @return a generated rate limiter name | ||||
|      */ | ||||
|     String generate(Object target, Method method, Object... args); | ||||
| } | ||||
| @@ -28,10 +28,12 @@ public enum LimitType { | ||||
|      * 全局限流 | ||||
|      */ | ||||
|     DEFAULT, | ||||
|  | ||||
|     /** | ||||
|      * 根据 IP 限流 | ||||
|      */ | ||||
|     IP, | ||||
|  | ||||
|     /** | ||||
|      * 根据实例限流(支持集群多实例) | ||||
|      */ | ||||
|   | ||||
| @@ -25,7 +25,12 @@ import top.continew.starter.core.exception.BaseException; | ||||
|  * @since 2.2.0 | ||||
|  */ | ||||
| public class RateLimiterException extends BaseException { | ||||
|  | ||||
|     public RateLimiterException(String message) { | ||||
|         super(message); | ||||
|     } | ||||
|  | ||||
|     public RateLimiterException(String message, Throwable cause) { | ||||
|         super(message, cause); | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -26,12 +26,5 @@ | ||||
|             <groupId>top.continew</groupId> | ||||
|             <artifactId>continew-starter-core</artifactId> | ||||
|         </dependency> | ||||
|  | ||||
|         <!-- Web 模块 --> | ||||
|         <dependency> | ||||
|             <groupId>top.continew</groupId> | ||||
|             <artifactId>continew-starter-web</artifactId> | ||||
|         </dependency> | ||||
|  | ||||
|     </dependencies> | ||||
| </project> | ||||
		Reference in New Issue
	
	Block a user