diff --git a/kernel-d-validator/validator-api/src/main/java/cn/stylefeng/roses/kernel/validator/context/RequestParamContext.java b/kernel-d-validator/validator-api/src/main/java/cn/stylefeng/roses/kernel/validator/context/RequestParamContext.java new file mode 100644 index 000000000..295ef1258 --- /dev/null +++ b/kernel-d-validator/validator-api/src/main/java/cn/stylefeng/roses/kernel/validator/context/RequestParamContext.java @@ -0,0 +1,65 @@ +package cn.stylefeng.roses.kernel.validator.context; + +import cn.hutool.core.lang.Dict; + +/** + * 临时保存http请求的参数 + *

+ * 可以保存@RequestBody的可以保存parameter方式传参的 + * + * @author fengshuonan + * @date 2020/8/20 + */ +public class RequestParamContext { + + private static final ThreadLocal CONTEXT_HOLDER = new ThreadLocal<>(); + + /** + * 保存请求参数 + * + * @author stylefeng + * @date 2020/6/21 20:17 + */ + public static void set(Dict requestParam) { + CONTEXT_HOLDER.set(requestParam); + } + + /** + * 保存请求参数 + * + * @author stylefeng + * @date 2020/6/21 20:17 + */ + public static void setObject(Object requestParam) { + + if (requestParam == null) { + return; + } + + if (requestParam instanceof Dict) { + CONTEXT_HOLDER.set((Dict) requestParam); + } else { + CONTEXT_HOLDER.set(Dict.parse(requestParam)); + } + } + + /** + * 获取请求参数 + * + * @author stylefeng + * @date 2020/6/21 20:17 + */ + public static Dict get() { + return CONTEXT_HOLDER.get(); + } + + /** + * 清除请求参数 + * + * @author stylefeng + * @date 2020/6/21 20:17 + */ + public static void clear() { + CONTEXT_HOLDER.remove(); + } +} diff --git a/kernel-d-validator/validator-api/src/main/java/cn/stylefeng/roses/kernel/validator/context/RequestParamIdContext.java b/kernel-d-validator/validator-api/src/main/java/cn/stylefeng/roses/kernel/validator/context/RequestParamIdContext.java deleted file mode 100644 index e8248c052..000000000 --- a/kernel-d-validator/validator-api/src/main/java/cn/stylefeng/roses/kernel/validator/context/RequestParamIdContext.java +++ /dev/null @@ -1,45 +0,0 @@ -package cn.stylefeng.roses.kernel.validator.context; - -/** - * 临时保存参数id字段值,用于唯一性校验 - *

- * 注意:如果要用@TableUniqueValue这个校验,必须得主键的字段名是id,否则会校验失败 - * - * @author fengshuonan - * @date 2020/11/4 14:34 - */ -public class RequestParamIdContext { - - private static final ThreadLocal PARAM_ID_HOLDER = new ThreadLocal<>(); - - /** - * 设置临时缓存的id - * - * @author fengshuonan - * @date 2020/11/4 14:35 - */ - public static void set(Long id) { - PARAM_ID_HOLDER.set(id); - } - - /** - * 获取临时缓存的id - * - * @author fengshuonan - * @date 2020/11/4 14:35 - */ - public static Long get() { - return PARAM_ID_HOLDER.get(); - } - - /** - * 清除临时缓存的id - * - * @author fengshuonan - * @date 2020/11/4 14:35 - */ - public static void clear() { - PARAM_ID_HOLDER.remove(); - } - -} diff --git a/kernel-d-validator/validator-api/src/main/java/cn/stylefeng/roses/kernel/validator/pojo/UniqueValidateParam.java b/kernel-d-validator/validator-api/src/main/java/cn/stylefeng/roses/kernel/validator/pojo/UniqueValidateParam.java index 044deada7..ac15d0db8 100644 --- a/kernel-d-validator/validator-api/src/main/java/cn/stylefeng/roses/kernel/validator/pojo/UniqueValidateParam.java +++ b/kernel-d-validator/validator-api/src/main/java/cn/stylefeng/roses/kernel/validator/pojo/UniqueValidateParam.java @@ -33,6 +33,11 @@ public class UniqueValidateParam { */ Boolean excludeCurrentRecord; + /** + * 主键id的字段名 + */ + String idFieldName; + /** * 当前记录的主键id */ diff --git a/kernel-d-validator/validator-api/src/main/java/cn/stylefeng/roses/kernel/validator/validators/unique/TableUniqueValue.java b/kernel-d-validator/validator-api/src/main/java/cn/stylefeng/roses/kernel/validator/validators/unique/TableUniqueValue.java index 84075b68c..6f867781c 100644 --- a/kernel-d-validator/validator-api/src/main/java/cn/stylefeng/roses/kernel/validator/validators/unique/TableUniqueValue.java +++ b/kernel-d-validator/validator-api/src/main/java/cn/stylefeng/roses/kernel/validator/validators/unique/TableUniqueValue.java @@ -38,6 +38,11 @@ public @interface TableUniqueValue { */ String columnName(); + /** + * 主键id的字段名,默认为字段名为:id + */ + String idFieldName() default "id"; + /** * 是否开启逻辑删除校验,默认是关闭的 *

diff --git a/kernel-d-validator/validator-api/src/main/java/cn/stylefeng/roses/kernel/validator/validators/unique/TableUniqueValueValidator.java b/kernel-d-validator/validator-api/src/main/java/cn/stylefeng/roses/kernel/validator/validators/unique/TableUniqueValueValidator.java index 8b4029886..159105cbe 100644 --- a/kernel-d-validator/validator-api/src/main/java/cn/stylefeng/roses/kernel/validator/validators/unique/TableUniqueValueValidator.java +++ b/kernel-d-validator/validator-api/src/main/java/cn/stylefeng/roses/kernel/validator/validators/unique/TableUniqueValueValidator.java @@ -1,11 +1,12 @@ package cn.stylefeng.roses.kernel.validator.validators.unique; +import cn.hutool.core.lang.Dict; import cn.hutool.core.util.ObjectUtil; import cn.stylefeng.roses.kernel.rule.pojo.request.BaseRequest; -import cn.stylefeng.roses.kernel.validator.validators.unique.service.TableUniqueValueService; import cn.stylefeng.roses.kernel.validator.context.RequestGroupContext; -import cn.stylefeng.roses.kernel.validator.context.RequestParamIdContext; +import cn.stylefeng.roses.kernel.validator.context.RequestParamContext; import cn.stylefeng.roses.kernel.validator.pojo.UniqueValidateParam; +import cn.stylefeng.roses.kernel.validator.validators.unique.service.TableUniqueValueService; import javax.validation.ConstraintValidator; import javax.validation.ConstraintValidatorContext; @@ -28,6 +29,11 @@ public class TableUniqueValueValidator implements ConstraintValidator @@ -54,6 +60,7 @@ public class TableUniqueValueValidator implements ConstraintValidator validateGroupClass = RequestGroupContext.get(); - // 如果属于add group,则校验库中所有行 - if (BaseRequest.add.class.equals(validateGroupClass)) { - UniqueValidateParam addParam = createAddParam(fieldValue); - return TableUniqueValueService.getFiledUniqueFlag(addParam); - } - // 如果属于edit group,校验时需要排除当前修改的这条记录 if (BaseRequest.edit.class.equals(validateGroupClass)) { UniqueValidateParam editParam = createEditParam(fieldValue); return TableUniqueValueService.getFiledUniqueFlag(editParam); } + // 如果属于add group,则校验库中所有行 + if (BaseRequest.add.class.equals(validateGroupClass)) { + UniqueValidateParam addParam = createAddParam(fieldValue); + return TableUniqueValueService.getFiledUniqueFlag(addParam); + } + // 默认校验所有的行 UniqueValidateParam addParam = createAddParam(fieldValue); return TableUniqueValueService.getFiledUniqueFlag(addParam); @@ -106,12 +113,17 @@ public class TableUniqueValueValidator implements ConstraintValidator " + uniqueValidateParam.getLogicDeleteValue() + ")", - uniqueValidateParam.getValue()); + String sqlTemplate = "select count(*) from {} where {} = {0} and ({} is null || {} <> {})"; + String finalSql = StrUtil.format(sqlTemplate, + uniqueValidateParam.getTableName(), + uniqueValidateParam.getColumnName(), + uniqueValidateParam.getLogicDeleteFieldName(), + uniqueValidateParam.getLogicDeleteFieldName(), + uniqueValidateParam.getLogicDeleteValue()); + resultCount = dbOperatorApi.selectCount(finalSql, uniqueValidateParam.getValue()); } // 排除当前记录,不排除逻辑删除的内容 @@ -58,11 +59,9 @@ public class TableUniqueValueService { // id判空 paramIdValidate(uniqueValidateParam); - resultCount = dbOperatorApi.selectCount( - "select count(*) from " + uniqueValidateParam.getTableName() - + " where " + uniqueValidateParam.getColumnName() + " = {0} " - + " and id <> {1}", - uniqueValidateParam.getValue(), uniqueValidateParam.getId()); + String sqlTemplate = "select count(*) from {} where {} = {0} and {} <> {1}"; + String finalSql = StrUtil.format(sqlTemplate, uniqueValidateParam.getTableName(), uniqueValidateParam.getColumnName(), uniqueValidateParam.getIdFieldName()); + resultCount = dbOperatorApi.selectCount(finalSql, uniqueValidateParam.getValue(), uniqueValidateParam.getId()); } // 排除当前记录,排除逻辑删除的内容 @@ -72,14 +71,15 @@ public class TableUniqueValueService { // id判空 paramIdValidate(uniqueValidateParam); - resultCount = dbOperatorApi.selectCount( - "select count(*) from " + uniqueValidateParam.getTableName() - + " where " + uniqueValidateParam.getColumnName() + " = {0} " - + " and id <> {1} " - + " and " - + "(" + uniqueValidateParam.getLogicDeleteFieldName() + " is null || " - + uniqueValidateParam.getLogicDeleteFieldName() + " <> " + uniqueValidateParam.getLogicDeleteValue() + ")", - uniqueValidateParam.getValue(), uniqueValidateParam.getId()); + String sqlTemplate = "select count(*) from {} where {} = {0} and {} <> {1} and ({} is null || {} <> {})"; + String finalSql = StrUtil.format(sqlTemplate, + uniqueValidateParam.getTableName(), + uniqueValidateParam.getColumnName(), + uniqueValidateParam.getIdFieldName(), + uniqueValidateParam.getLogicDeleteFieldName(), + uniqueValidateParam.getLogicDeleteFieldName(), + uniqueValidateParam.getLogicDeleteValue()); + resultCount = dbOperatorApi.selectCount(finalSql, uniqueValidateParam.getValue(), uniqueValidateParam.getId()); } // 如果大于0,代表不是唯一的当前校验的值 diff --git a/kernel-d-validator/validator-spring-boot-starter/src/main/java/cn/stylefeng/roses/kernel/validator/starter/GunsValidatorAutoConfiguration.java b/kernel-d-validator/validator-spring-boot-starter/src/main/java/cn/stylefeng/roses/kernel/validator/starter/ValidatorAutoConfiguration.java similarity index 98% rename from kernel-d-validator/validator-spring-boot-starter/src/main/java/cn/stylefeng/roses/kernel/validator/starter/GunsValidatorAutoConfiguration.java rename to kernel-d-validator/validator-spring-boot-starter/src/main/java/cn/stylefeng/roses/kernel/validator/starter/ValidatorAutoConfiguration.java index 598e6220b..2ed1254a9 100644 --- a/kernel-d-validator/validator-spring-boot-starter/src/main/java/cn/stylefeng/roses/kernel/validator/starter/GunsValidatorAutoConfiguration.java +++ b/kernel-d-validator/validator-spring-boot-starter/src/main/java/cn/stylefeng/roses/kernel/validator/starter/ValidatorAutoConfiguration.java @@ -22,7 +22,7 @@ import org.springframework.context.annotation.Configuration; * @date 2020/12/1 21:44 */ @Configuration -public class GunsValidatorAutoConfiguration { +public class ValidatorAutoConfiguration { /** * 黑名单校验 diff --git a/kernel-d-validator/validator-spring-boot-starter/src/main/java/cn/stylefeng/roses/kernel/validator/starter/resolver/MethodArgumentResolver.java b/kernel-d-validator/validator-spring-boot-starter/src/main/java/cn/stylefeng/roses/kernel/validator/starter/resolver/MethodArgumentResolver.java new file mode 100644 index 000000000..7d5346a2c --- /dev/null +++ b/kernel-d-validator/validator-spring-boot-starter/src/main/java/cn/stylefeng/roses/kernel/validator/starter/resolver/MethodArgumentResolver.java @@ -0,0 +1,53 @@ +package cn.stylefeng.roses.kernel.validator.starter.resolver; + +import cn.stylefeng.roses.kernel.validator.starter.web.GunsValidator; +import cn.stylefeng.roses.kernel.validator.starter.web.ValidatorRequestResponseBodyMethodProcessor; +import org.springframework.context.annotation.Bean; +import org.springframework.context.annotation.Configuration; +import org.springframework.web.method.support.HandlerMethodArgumentResolver; +import org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter; + +import javax.annotation.PostConstruct; +import javax.annotation.Resource; +import java.util.ArrayList; +import java.util.List; +import java.util.Objects; + +/** + * 自定义的GunsRequestResponseBodyMethodProcessor,放在所有resolvers之前 + * + * @author fengshuonan + * @date 2020/8/21 21:09 + */ +@Configuration +public class MethodArgumentResolver { + + @Resource + private RequestMappingHandlerAdapter adapter; + + /** + * 自定义的spring参数校验器,重写主要为了保存一些在自定义validator中读不到的属性 + * + * @author fengshuonan + * @date 2020/8/12 20:18 + */ + @Bean + public GunsValidator gunsValidator() { + return new GunsValidator(); + } + + /** + * 自定义的GunsRequestResponseBodyMethodProcessor,放在所有resolvers之前 + * + * @author fengshuonan + * @date 2020/12/16 18:34 + */ + @PostConstruct + public void injectSelfMethodArgumentResolver() { + List argumentResolvers = new ArrayList<>(); + argumentResolvers.add(new ValidatorRequestResponseBodyMethodProcessor(adapter.getMessageConverters())); + argumentResolvers.addAll(Objects.requireNonNull(adapter.getArgumentResolvers())); + adapter.setArgumentResolvers(argumentResolvers); + } + +} diff --git a/kernel-d-validator/validator-spring-boot-starter/src/main/java/cn/stylefeng/roses/kernel/validator/starter/web/GunsValidator.java b/kernel-d-validator/validator-spring-boot-starter/src/main/java/cn/stylefeng/roses/kernel/validator/starter/web/GunsValidator.java new file mode 100644 index 000000000..5823aad8c --- /dev/null +++ b/kernel-d-validator/validator-spring-boot-starter/src/main/java/cn/stylefeng/roses/kernel/validator/starter/web/GunsValidator.java @@ -0,0 +1,38 @@ +package cn.stylefeng.roses.kernel.validator.starter.web; + +import cn.stylefeng.roses.kernel.validator.context.RequestGroupContext; +import lombok.extern.slf4j.Slf4j; +import org.springframework.validation.Errors; +import org.springframework.validation.beanvalidation.LocalValidatorFactoryBean; + +/** + * 用于真正校验参数之前缓存一下group的class类型 + *

+ * 因为ConstraintValidator的自定义校验中获取不到当前进行的group + * + * @author fengshuonan + * @date 2020/8/12 20:07 + */ +@Slf4j +public class GunsValidator extends LocalValidatorFactoryBean { + + @Override + public void validate(Object target, Errors errors, Object... validationHints) { + + try { + if (validationHints.length > 0) { + + // 如果是class类型,利用ThreadLocal缓存一下class类型 + if (validationHints[0] instanceof Class) { + + // 临时保存group的class值 + RequestGroupContext.set((Class) validationHints[0]); + } + } + super.validate(target, errors, validationHints); + } finally { + RequestGroupContext.clear(); + } + } + +} diff --git a/kernel-d-validator/validator-spring-boot-starter/src/main/java/cn/stylefeng/roses/kernel/validator/starter/web/ValidatorRequestResponseBodyMethodProcessor.java b/kernel-d-validator/validator-spring-boot-starter/src/main/java/cn/stylefeng/roses/kernel/validator/starter/web/ValidatorRequestResponseBodyMethodProcessor.java new file mode 100644 index 000000000..b266dbfc9 --- /dev/null +++ b/kernel-d-validator/validator-spring-boot-starter/src/main/java/cn/stylefeng/roses/kernel/validator/starter/web/ValidatorRequestResponseBodyMethodProcessor.java @@ -0,0 +1,70 @@ +package cn.stylefeng.roses.kernel.validator.starter.web; + +import cn.hutool.core.util.StrUtil; +import cn.stylefeng.roses.kernel.validator.context.RequestParamContext; +import cn.stylefeng.roses.kernel.validator.exception.ParamValidateException; +import org.springframework.core.Conventions; +import org.springframework.core.MethodParameter; +import org.springframework.http.converter.HttpMessageConverter; +import org.springframework.lang.Nullable; +import org.springframework.validation.BindingResult; +import org.springframework.validation.ObjectError; +import org.springframework.web.bind.WebDataBinder; +import org.springframework.web.bind.support.WebDataBinderFactory; +import org.springframework.web.context.request.NativeWebRequest; +import org.springframework.web.method.support.ModelAndViewContainer; +import org.springframework.web.servlet.mvc.method.annotation.RequestResponseBodyMethodProcessor; + +import java.util.List; + +import static cn.stylefeng.roses.kernel.validator.exception.enums.ValidatorExceptionEnum.PARAM_VALIDATE_ERROR; + +/** + * 拓展原有RequestResponseBodyMethodProcessor,只为缓存临时参数 + * + * @author fengshuonan + * @date 2020/8/21 20:51 + */ +public class ValidatorRequestResponseBodyMethodProcessor extends RequestResponseBodyMethodProcessor { + + public ValidatorRequestResponseBodyMethodProcessor(List> converters) { + super(converters); + } + + @Override + public Object resolveArgument(MethodParameter parameter, @Nullable ModelAndViewContainer mavContainer, NativeWebRequest webRequest, @Nullable WebDataBinderFactory binderFactory) throws Exception { + + parameter = parameter.nestedIfOptional(); + Object arg = readWithMessageConverters(webRequest, parameter, parameter.getNestedGenericParameterType()); + + // 临时缓存一下@RequestBody注解上的参数 + RequestParamContext.setObject(arg); + + String name = Conventions.getVariableNameForParameter(parameter); + + if (binderFactory != null) { + WebDataBinder binder = binderFactory.createBinder(webRequest, arg, name); + if (arg != null) { + validateIfApplicable(binder, parameter); + if (binder.getBindingResult().hasErrors() && isBindExceptionRequired(binder, parameter)) { + List allErrors = binder.getBindingResult().getAllErrors(); + StringBuilder errTips = new StringBuilder(); + int index = 1; + for (ObjectError error : allErrors) { + errTips.append(index++); + errTips.append("."); + errTips.append(error.getDefaultMessage()); + errTips.append(";"); + } + String userTip = StrUtil.format(PARAM_VALIDATE_ERROR.getUserTip(), errTips.toString()); + throw new ParamValidateException(PARAM_VALIDATE_ERROR, userTip); + } + } + if (mavContainer != null) { + mavContainer.addAttribute(BindingResult.MODEL_KEY_PREFIX + name, binder.getBindingResult()); + } + } + + return adaptArgumentIfNecessary(arg, parameter); + } +} diff --git a/kernel-d-validator/validator-spring-boot-starter/src/main/resources/META-INF/spring.factories b/kernel-d-validator/validator-spring-boot-starter/src/main/resources/META-INF/spring.factories index 6632a9974..0168033ff 100644 --- a/kernel-d-validator/validator-spring-boot-starter/src/main/resources/META-INF/spring.factories +++ b/kernel-d-validator/validator-spring-boot-starter/src/main/resources/META-INF/spring.factories @@ -1,2 +1,3 @@ org.springframework.boot.autoconfigure.EnableAutoConfiguration=\ - cn.stylefeng.roses.kernel.validator.starter.GunsValidatorAutoConfiguration \ No newline at end of file +cn.stylefeng.roses.kernel.validator.starter.ValidatorAutoConfiguration,\ +cn.stylefeng.roses.kernel.validator.starter.resolver.MethodArgumentResolver \ No newline at end of file