Solon详解系列文章:

Solon详解(一)- 快速入门

Solon详解(二)- Solon的核心

Solon详解(三)- Solon的web开发

Solon详解(四)- Solon的事务传播机制

Solon详解(五)- Solon扩展机制之Solon Plugin

Solon详解(六)- Solon的校验扩展框架使用与扩展

在业务的实现过程中,尤其是对外接口开发,我们需要对请求进行大量的验证并返回错误状态码和描述。lombok 框架有很多很赞的注解,但是人家是throw一个异常,这与有些需求不一定能匹配。

该文将介绍Solon的扩展验证框架:solon.extend.validation 的使用和扩展。效果如下:


  1. @XValid
  2. @XController
  3. public class UserController extends VerifyController{
  4. @RepeatSubmit //重复提交验证
  5. @Whitelist //IP白名单验证
  6. @NotNull({"name", "mobile", "icon", "code"}) //非NULL验证
  7. @Numeric({"code"})
  8. @XMapping("/user/add")
  9. public void addUser(String name, @Pattern("^http") String icon, int code, @Pattern("^13\\d{9}$") String mobile){
  10. //...
  11. }
  12. }

相较于 Spring 的 Validator 是争对 Bean,Solon 则是争对 XContext(即http参数)。这点区别非常大,Solon 是在 XAction 执行之前对 http 参数进行校验。

注解 作用范围 说明
Date 参数 校验注解的参数值为日期格式
DecimalMax(value) 参数 校验注解的参数值小于等于@ DecimalMax指定的value值
DecimalMin(value) 参数 校验注解的参数值大于等于@ DecimalMin指定的value值
Email 参数 校验注解的参数值为电子邮箱格式
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 默认使用了本地延时锁。如果是分布式环境,需要定制为分布式锁:

  1. public class NoRepeatLockNew implements NoRepeatLock {
  2. @Override
  3. public boolean tryLock(String key, int seconds) {
  4. //使用分布式锁
  5. //
  6. return LockUtils.tryLock(XWaterAdapter.global().service_name(), key, seconds);
  7. }
  8. }
  9. ValidatorManager.setNoRepeatLock(new NoRepeatLockNew());

2、@Whitelist 实现验证

框架层面没办法为 Whitelist 提供一个名单库,所以需要通过一个接口实现完成对接。

  1. public class WhitelistCheckerNew implements WhitelistChecker {
  2. @Override
  3. public boolean check(Whitelist anno, XContext ctx) {
  4. String ip = IPUtils.getIP(ctx);
  5. return WaterClient.Whitelist.existsOfServerIp(ip);
  6. }
  7. }
  8. ValidatorManager.setWhitelistChecker(new WhitelistCheckerNew());

3、改造校验输出

solon.extend.validation 默认输出 http 400 状态 + json;尝试改改去掉 http 400 状态。

  1. @XConfiguration
  2. public class Config {
  3. @XBean //Solon 的 @XBean 也支持空函数,为其它提运行申明
  4. public void adapter() {
  5. ValidatorManager.globalSet(new ValidatorManager((ctx, ano, rst, message) -> {
  6. ctx.setHandled(true);
  7. if (XUtil.isEmpty(message)) {
  8. message = new StringBuilder(100)
  9. .append("@")
  10. .append(ano.annotationType().getSimpleName())
  11. .append(" verification failed")
  12. .toString();
  13. }
  14. ctx.output(message);
  15. return true;
  16. }));
  17. }
  18. }

二、添一个扩展注解

1、先定义个校验注解 @Date

偷懒一下,直接把自带的扔出来了。只要看着能自己搞就行了:-P

  1. @Target({ElementType.PARAMETER}) //只让它作用到参数,不管作用在哪,最终都是对XContext的校验
  2. @Retention(RetentionPolicy.RUNTIME)
  3. public @interface Date {
  4. @XNote("日期表达式, 默认为:ISO格式") //用XNote注解,是为了用时还能看到这个注释
  5. String value() default "";
  6. String message() default "";
  7. }

2、添加 @Date 的校验器实现类

  1. public class DateValidator implements Validator<Date> {
  2. public static final DateValidator instance = new DateValidator();
  3. @Override
  4. public String message(Date anno) {
  5. return anno.message();
  6. }
  7. @Override
  8. public XResult validate(XContext ctx, Date anno, String name, StringBuilder tmp) {
  9. String val = ctx.param(name);
  10. if (val == null || tryParse(anno, val) == false) {
  11. tmp.append(',').append(name);
  12. }
  13. if (tmp.length() > 1) {
  14. return XResult.failure(tmp.substring(1));
  15. } else {
  16. return XResult.succeed();
  17. }
  18. }
  19. private boolean tryParse(Date anno, String val) {
  20. try {
  21. if (XUtil.isEmpty(anno.value())) {
  22. DateTimeFormatter.ISO_LOCAL_DATE_TIME.parse(val);
  23. } else {
  24. DateTimeFormatter.ofPattern(anno.value()).parse(val);
  25. }
  26. return true;
  27. } catch (Exception ex) {
  28. return false;
  29. }
  30. }
  31. }

3、注册到校验管理器

  1. @XConfiguration
  2. public class Config {
  3. @XBean
  4. public void adapter() {
  5. ValidatorManager.global().register(Date.class, DateValidator.instance);
  6. }
  7. }

4、使用一下

  1. @XValid
  2. @XController
  3. public class UserController extends VerifyController{
  4. @XMapping("/user/add")
  5. public void addUser(String name, @Date("yyyy-MM-dd") String birthday){
  6. //...
  7. }
  8. }

Solon详解(六)- Solon的校验扩展框架使用与扩展的更多相关文章

  1. Springboot mini - Solon详解(六)- Solon的校验框架使用、定制与扩展

    Springboot min -Solon 详解系列文章: Springboot mini - Solon详解(一)- 快速入门 Springboot mini - Solon详解(二)- Solon ...

  2. Springboot mini - Solon详解(五)- Solon扩展机制之Solon Plugin

    Springboot min -Solon 详解系列文章: Springboot mini - Solon详解(一)- 快速入门 Springboot mini - Solon详解(二)- Solon ...

  3. Solon详解(二)- Solon的核心

    Solon详解系列文章: Solon详解(一)- 快速入门 Solon详解(二)- Solon的核心 Solon详解(三)- Solon的web开发 Solon详解(四)- Solon的事务传播机制 ...

  4. Solon详解(三)- Solon的web开发

    Solon详解系列文章: Solon详解(一)- 快速入门 Solon详解(二)- Solon的核心 Solon详解(三)- Solon的web开发 Solon详解(四)- Solon的事务传播机制 ...

  5. Solon详解(七)- Solon Ioc 的注解对比Spring及JSR330

    Solon详解系列文章: Solon详解(一)- 快速入门 Solon详解(二)- Solon的核心 Solon详解(三)- Solon的web开发 Solon详解(四)- Solon的事务传播机制 ...

  6. Solon详解(八)- Solon的缓存框架使用和定制

    Solon详解系列文章: Solon详解(一)- 快速入门 Solon详解(二)- Solon的核心 Solon详解(三)- Solon的web开发 Solon详解(四)- Solon的事务传播机制 ...

  7. Solon详解(九)- 渲染控制之定制统一的接口输出

    Solon详解系列文章: Solon详解(一)- 快速入门 Solon详解(二)- Solon的核心 Solon详解(三)- Solon的web开发 Solon详解(四)- Solon的事务传播机制 ...

  8. Solon详解(十)- 怎么用 Solon 开发基于 undertow jsp tld 的项目?

    Solon详解系列文章: Solon详解(一)- 快速入门 Solon详解(二)- Solon的核心 Solon详解(三)- Solon的web开发 Solon详解(四)- Solon的事务传播机制 ...

  9. Solon详解(11)- Mybatis 与 Solon 相亲相爱

    Solon详解系列文章: Solon详解(一)- 快速入门 Solon详解(二)- Solon的核心 Solon详解(三)- Solon的web开发 Solon详解(四)- Solon的事务传播机制 ...

随机推荐

  1. js 自定义属性操作

    自定义属性操作     element.属性 获取内置属性值     element.getAttribute("属性") 我们自己添加的属性叫自定义属性     element. ...

  2. C#LeetCode刷题之#191-位1的个数(Number of 1 Bits)

    问题 该文章的最新版本已迁移至个人博客[比特飞],单击链接 https://www.byteflying.com/archives/4052 访问. 编写一个函数,输入是一个无符号整数,返回其二进制表 ...

  3. Android开发进程0.1 轮播图 Scrollview Fragment

    轮播图的实现 轮播图通过banner可以较为便捷的实现 1.添加本地依赖,在dependence中搜索相关依赖 2.添加banner的view组件 3.创建适配器GlideImageLoader ex ...

  4. CentOS7(Linux)源码安装MySQL5.7.X

    介绍 软件应用最重要的就是数据库了,可是还有小伙伴不会在Linux上安装MySQL数据库,今天就来讲讲如何在CentOS7环境使用源码进行安装MySQL5.7.X. MySQL官网下载链接:https ...

  5. Kubernetes实战指南(三十三):都0202了,你还在手写k8s的yaml文件?

    目录 1. k8s的yaml文件到底有多复杂 2. 基于图形化的方式自动生成yaml 2.1 k8s图形化管理工具Ratel安装 2.2 使用Ratel创建生成yaml文件 2.2.1 基本配置 2. ...

  6. 远程控制(远控Bin)

    一.概念 控制端:运行在攻击者的电脑中,负责控制其他肉鸡: 被控制端:需要去生成,运行在肉鸡上,木马病毒 二.使用远控Bin工具实现内网远程控制 Step1:打开Kill.exe,点击系统设置,修改监 ...

  7. 笔记:CSS基础

    一.CSS(层叠式样式表),决定页面怎么显示元素 1.引入方式: 行内样式,在当前标签元素中直接使用 style 的属性. 内嵌方式,在<head>中写样式: 外链式,<link&g ...

  8. PC,移动端H5实现实现小球加入购物车效果

    HTML部分: <!DOCTYPE html> <html> <head> <meta http-equiv="content-type" ...

  9. SpringSecurity权限管理系统实战—八、AOP 记录用户、异常日志

    目录 SpringSecurity权限管理系统实战-一.项目简介和开发环境准备 SpringSecurity权限管理系统实战-二.日志.接口文档等实现 SpringSecurity权限管理系统实战-三 ...

  10. ES日期存储

    前段时间,在用ES时候,关于时间类型遇见点坑,ES版本是5.6.9,.但是在进行存储的时候"2019/01/01"或者是"2019/01/01 11:11:11" ...