Solon详解(六)- Solon的校验扩展框架使用与扩展
Solon详解系列文章:
Solon详解(一)- 快速入门
Solon详解(二)- Solon的核心
Solon详解(三)- Solon的web开发
Solon详解(四)- Solon的事务传播机制
Solon详解(五)- Solon扩展机制之Solon Plugin
Solon详解(六)- Solon的校验扩展框架使用与扩展
在业务的实现过程中,尤其是对外接口开发,我们需要对请求进行大量的验证并返回错误状态码和描述。lombok 框架有很多很赞的注解,但是人家是throw一个异常,这与有些需求不一定能匹配。
该文将介绍Solon的扩展验证框架:solon.extend.validation
的使用和扩展。效果如下:
@XValid
@XController
public class UserController extends VerifyController{
@RepeatSubmit //重复提交验证
@Whitelist //IP白名单验证
@NotNull({"name", "mobile", "icon", "code"}) //非NULL验证
@Numeric({"code"})
@XMapping("/user/add")
public void addUser(String name, @Pattern("^http") String icon, int code, @Pattern("^13\\d{9}$") String mobile){
//...
}
}
相较于 Spring 的 Validator 是争对 Bean,Solon 则是争对 XContext(即http参数)。这点区别非常大,Solon 是在 XAction 执行之前对 http 参数进行校验。
注解 | 作用范围 | 说明 |
---|---|---|
Date | 参数 | 校验注解的参数值为日期格式 |
DecimalMax(value) | 参数 | 校验注解的参数值小于等于@ DecimalMax指定的value值 |
DecimalMin(value) | 参数 | 校验注解的参数值大于等于@ DecimalMin指定的value值 |
参数 | 校验注解的参数值为电子邮箱格式 | |
Length(min, max) | 参数 | 校验注解的参数值长度在min和max区间内 |
Max(value) | 参数 | 校验注解的参数值小于等于@Max指定的value值 |
Min(value) | 参数 | 校验注解的参数值大于等于@Min指定的value值 |
NoRepeatSubmit | 控制器 或 动作 | 校验本次请求没有重复 |
NotBlank | 动作 或 参数 | 校验注解的参数值不是空白 |
NotEmpty | 动作 或 参数 | 校验注解的参数值不是空 |
NotNull | 动作 或 参数 | 校验注解的参数值不是null |
NotZero | 动作 或 参数 | 校验注解的参数值不是0 |
Null | 动作 或 参数 | 校验注解的参数值是null |
Numeric | 动作 或 参数 | 校验注解的参数值为数字格式 |
Pattern(value) | 参数 | 校验注解的参数值与指定的正则表达式匹配 |
Whitelist | 控制器 或 动作 | 校验本次请求在白名单范围内 |
XValid | 控制器 或 动作 | 为控制器 或 动作启用验证能力 |
可作用在 [动作 或 参数] 上的注解,加在动作上时可支持多个参数的校验。
一、定制使用
solon.extend.validation 通过 ValidatorManager,提供了一组定制和扩展接口。
1、@NoRepeatSubmit 改为分布式锁
NoRepeatSubmit 默认使用了本地延时锁。如果是分布式环境,需要定制为分布式锁:
public class NoRepeatLockNew implements NoRepeatLock {
@Override
public boolean tryLock(String key, int seconds) {
//使用分布式锁
//
return LockUtils.tryLock(XWaterAdapter.global().service_name(), key, seconds);
}
}
ValidatorManager.setNoRepeatLock(new NoRepeatLockNew());
2、@Whitelist 实现验证
框架层面没办法为 Whitelist 提供一个名单库,所以需要通过一个接口实现完成对接。
public class WhitelistCheckerNew implements WhitelistChecker {
@Override
public boolean check(Whitelist anno, XContext ctx) {
String ip = IPUtils.getIP(ctx);
return WaterClient.Whitelist.existsOfServerIp(ip);
}
}
ValidatorManager.setWhitelistChecker(new WhitelistCheckerNew());
3、改造校验输出
solon.extend.validation 默认输出 http 400 状态 + json;尝试改改去掉 http 400 状态。
@XConfiguration
public class Config {
@XBean //Solon 的 @XBean 也支持空函数,为其它提运行申明
public void adapter() {
ValidatorManager.globalSet(new ValidatorManager((ctx, ano, rst, message) -> {
ctx.setHandled(true);
if (XUtil.isEmpty(message)) {
message = new StringBuilder(100)
.append("@")
.append(ano.annotationType().getSimpleName())
.append(" verification failed")
.toString();
}
ctx.output(message);
return true;
}));
}
}
二、添一个扩展注解
1、先定义个校验注解 @Date
偷懒一下,直接把自带的扔出来了。只要看着能自己搞就行了:-P
@Target({ElementType.PARAMETER}) //只让它作用到参数,不管作用在哪,最终都是对XContext的校验
@Retention(RetentionPolicy.RUNTIME)
public @interface Date {
@XNote("日期表达式, 默认为:ISO格式") //用XNote注解,是为了用时还能看到这个注释
String value() default "";
String message() default "";
}
2、添加 @Date 的校验器实现类
public class DateValidator implements Validator<Date> {
public static final DateValidator instance = new DateValidator();
@Override
public String message(Date anno) {
return anno.message();
}
@Override
public XResult validate(XContext ctx, Date anno, String name, StringBuilder tmp) {
String val = ctx.param(name);
if (val == null || tryParse(anno, val) == false) {
tmp.append(',').append(name);
}
if (tmp.length() > 1) {
return XResult.failure(tmp.substring(1));
} else {
return XResult.succeed();
}
}
private boolean tryParse(Date anno, String val) {
try {
if (XUtil.isEmpty(anno.value())) {
DateTimeFormatter.ISO_LOCAL_DATE_TIME.parse(val);
} else {
DateTimeFormatter.ofPattern(anno.value()).parse(val);
}
return true;
} catch (Exception ex) {
return false;
}
}
}
3、注册到校验管理器
@XConfiguration
public class Config {
@XBean
public void adapter() {
ValidatorManager.global().register(Date.class, DateValidator.instance);
}
}
4、使用一下
@XValid
@XController
public class UserController extends VerifyController{
@XMapping("/user/add")
public void addUser(String name, @Date("yyyy-MM-dd") String birthday){
//...
}
}
Solon详解(六)- Solon的校验扩展框架使用与扩展的更多相关文章
- Springboot mini - Solon详解(六)- Solon的校验框架使用、定制与扩展
Springboot min -Solon 详解系列文章: Springboot mini - Solon详解(一)- 快速入门 Springboot mini - Solon详解(二)- Solon ...
- Springboot mini - Solon详解(五)- Solon扩展机制之Solon Plugin
Springboot min -Solon 详解系列文章: Springboot mini - Solon详解(一)- 快速入门 Springboot mini - Solon详解(二)- Solon ...
- Solon详解(二)- Solon的核心
Solon详解系列文章: Solon详解(一)- 快速入门 Solon详解(二)- Solon的核心 Solon详解(三)- Solon的web开发 Solon详解(四)- Solon的事务传播机制 ...
- Solon详解(三)- Solon的web开发
Solon详解系列文章: Solon详解(一)- 快速入门 Solon详解(二)- Solon的核心 Solon详解(三)- Solon的web开发 Solon详解(四)- Solon的事务传播机制 ...
- Solon详解(七)- Solon Ioc 的注解对比Spring及JSR330
Solon详解系列文章: Solon详解(一)- 快速入门 Solon详解(二)- Solon的核心 Solon详解(三)- Solon的web开发 Solon详解(四)- Solon的事务传播机制 ...
- Solon详解(八)- Solon的缓存框架使用和定制
Solon详解系列文章: Solon详解(一)- 快速入门 Solon详解(二)- Solon的核心 Solon详解(三)- Solon的web开发 Solon详解(四)- Solon的事务传播机制 ...
- Solon详解(九)- 渲染控制之定制统一的接口输出
Solon详解系列文章: Solon详解(一)- 快速入门 Solon详解(二)- Solon的核心 Solon详解(三)- Solon的web开发 Solon详解(四)- Solon的事务传播机制 ...
- Solon详解(十)- 怎么用 Solon 开发基于 undertow jsp tld 的项目?
Solon详解系列文章: Solon详解(一)- 快速入门 Solon详解(二)- Solon的核心 Solon详解(三)- Solon的web开发 Solon详解(四)- Solon的事务传播机制 ...
- Solon详解(11)- Mybatis 与 Solon 相亲相爱
Solon详解系列文章: Solon详解(一)- 快速入门 Solon详解(二)- Solon的核心 Solon详解(三)- Solon的web开发 Solon详解(四)- Solon的事务传播机制 ...
随机推荐
- Python 3.9 beta2 版本发布了,看看新特性?
随着 Python 3.9.0b1 的发布,即开发周期中计划的四个 beta 版本的首个,Python 3.9 的功能已经是完善了.在 10 月发布最终版本之前,还会有许多测试和稳定性方面的工作要做. ...
- CSS高级特效(上)
1.CSS Shapes CSS Shapes是一个新标准,旨在让Web设计者能使用各种形状. CSS Shapes包含两组新属性,一组用于设置影响盒子中内容的形状,另一组用于设置影响形状元素周边内容 ...
- [vue]子组件通过props获取父组件数据以及使用watch解决动态数据不生效问题
父子组件的传值,在Vue官方也写得很清楚,父组件中使用v-bind绑定传送,子组件使用props接收. 父组件通过v-bind绑定数据: <template> <router-vie ...
- C#LeetCode刷题之#463-岛屿的周长(Island Perimeter)
问题 该文章的最新版本已迁移至个人博客[比特飞],单击链接 https://www.byteflying.com/archives/3794 访问. 给定一个包含 0 和 1 的二维网格地图,其中 1 ...
- 从一次外卖到对oauth2.0的思考
别问oauth1.0哪去了,问就是不好讲. 1. 外卖并不好吃 今天下班得早,想吃顿好的,于是就点了一份外卖,过了一会儿,外卖到了,但是在小区大门被堵住了,我亲自远程开门后才能进来,又过了一会,被 ...
- SOAR安全能力编排化
安全能力编排化(Security Capability Orchestration)是指系统一方面可以通过自底向上地通过安全设施接口化和安全接口应用化实现安全应用编排化:另一方面则自顶向下地将安全运营 ...
- Solon 的 PathVariable 不需注解
相对于 Spring boot 的 path variable :Solon 的不需注解,只要变量名对上即可: //path var demo // @XMapping("e/{p_q}/{ ...
- 微信小程序扫码解析小程序码
通过微信扫小程序码,跳转到应用小程序内, 如何解析小程序码的参数呢? 一般小程序码会跳转到设置的页面,如首页, 可以直接跳转到小程序首页,然后解析小程序携带的参数,再打开某个页面. (小程序码的路径要 ...
- StructuredStreaming(New)
SparkStreaming API using DataSets and DataFrames (New) 使用流式DataSets和流式DataFrames的API ◆ 1.创建流式DataFr ...
- Flink的应用场景和架构
Flink的应用场景 Flink项目的理念就是:Flink是为分布式,高性能,随时可用以及准确的流处理应用程序打造的开源流处理框架.自2019年开源以来,迅速成为大数据实时计算领域炙手可热的技术框架. ...