DDD-声明式复杂规则校验实践

当进行DDD编程过程中,有个非常繁琐,但又是非常重要的步骤:整理出某个实体的所有业务字段约束。

关于业务校验,业界已经有一些框架支持,如validator框架。但大部分人只用于参数校验,并未将其用于实体内。

本人整理出了一种声明式表达业务的方式。写起约束来个人感觉比较聚焦,从而可以辅助思考和整理约束。

直接上示例代码:
以下代码体现了一个品牌的业务约束和行为操作。

  • BrandNameDuplicateChecker:展示针对实体的复杂校验
  • BrandLogoChecker:展示针对实体单个字段的复杂校验
  • Spec:复杂校验注解声明,内部类SpecChecker是关键代码

品牌实体

@Data
@Setter(AccessLevel.PROTECTED)
@Accessors(chain = true)
@Spec(value = BrandNameDuplicateChecker.class) //复杂规则校验,一般涉及bean依赖的校验
public class Brand extends BaseEntity<Brand> {

  @ApiModelProperty("品牌ID")
  private Long id;

  @Size(min=1, max=100, message = "品牌名称字符数限制为[1,100]")
  @ApiModelProperty("品牌名称")
  private String name;

  @Spec(value = BrandLogoChecker.class)
  @Pattern(regexp = "(https?|ftp|file)://[-A-Za-z0-9+&@#/%?=~_|!:,.;]+[-A-Za-z0-9+&@#/%=~_|]", message = "logo必须为合法的URL")
  @Size(min=1, max = 128, message = "品牌名称长度必须在1-128字符内")
  @ApiModelProperty("品牌logo,为标准的URL图片格式")
  private String logo;

  @ApiModelProperty("排序分值")
  private Long score;

  @ApiModelProperty("是否隐藏")
  private Boolean hidden;

  @ApiModelProperty("版本号")
  private Long version;
  @ApiModelProperty("创建时间")
  private Date createdAt;
  @ApiModelProperty("更新时间")
  private Date updatedAt;

  protected Brand(){}
  
  
  public Brand(Long id, String name, String logo, Long score, Boolean hidden) {
    this.id = id;
    this.name = name;
    this.logo = logo;
    this.score = score;
    this.hidden = hidden;
    
    version = 0L;
    createdAt = new Date();
    updatedAt = new Date();
    
    this.validate(); //触发校验
  }
  
  public Brand changeName(String name){
    
    Brand toUpdate = new Brand().setId(this.id).setName(name);
    toUpdate.validate("name");
    DomainRegistry.bean(BrandNameDuplicateChecker.class).check(toUpdate);
    
    this.name = name;
    return this;
  }
  
  public Brand changeLogo(String logo){
    
    Brand toUpdate = new Brand().setId(this.id).setLogo(logo);
    toUpdate.validate("logo");
    
    this.logo = logo;
    return this;
  }
  
  public Brand changeScore(Long score){
    
    Brand toUpdate = new Brand().setId(this.id).setScore(score);
    toUpdate.validate("score");
    
    this.score = score;
    return this;
  }
  
  public Brand hidden(){
    
    this.hidden = true;
    return this;
  }
  
  public Brand visible(){
    
    this.hidden = false;
    return this;
  }

  public static Brand load(Long id){
    return Optional.ofNullable(DomainRegistry.repo(BrandRepo.class).findById(id))
        .orElseThrow(()->new BusinessException("品牌不存在"));
  }
}

针对实体本身的复杂校验

@Component
@Slf4j
public class BrandNameDuplicateChecker implements Checker<Brand>{
  
  @Autowired
  @Setter
  private BrandRepo brandRepo;
  
  @Override
  public void check(Brand brand) {
    if(Strings.isNullOrEmpty(brand.getName())){
      return;
    }
    Brand exist = brandRepo.findByName(brand.getName());
    if(exist!=null && ! Objects.equals(brand.getId(), exist.getId())){
      throw new BusinessException("品牌名称已经被使用");
    }
  }
}

针对实体字段的复杂校验

@Component
public class BrandLogoChecker implements Checker<String> {
  
  @Autowired
  @Setter
  private PhotoUrlChecker photoUrlChecker;
  
  @Override
  public void check(String logo) {
    if(photoUrlChecker.invalid(logo)){
      throw new BusinessException("品牌Logo的URL不合法");
    }
  }
}

复杂校验实现

@Target({ ElementType.FIELD, ElementType.METHOD, ElementType.TYPE })
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Constraint(validatedBy = { Spec.SpecChecker.class })
public @interface Spec {
  
  Class<? extends Checker>[] value();
  
  String message() default "字段不符合条件约束";
  
  Class<?>[] groups() default {};
  
  Class<? extends Payload>[] payload() default {};
  
  
  @Slf4j
  public static class SpecChecker implements ConstraintValidator<Spec, Object> {
    
    Spec annotation;
    
    List<Class<? extends Checker>> checkerClasses;
  
    Map<Class<? extends Checker>, ? extends Checker> allCheckers;
    
    @Override
    public void initialize(Spec constraintAnnotation) {
      annotation = constraintAnnotation;
      checkerClasses = Arrays.asList(annotation.value());
      allCheckers = DomainRegistry.beanMap(Checker.class)
          .values().stream()
          .collect(Collectors.toMap(Checker::getClass, Function.identity()));
    }
    
    @Override
    public boolean isValid(Object object, ConstraintValidatorContext constraintValidatorContext) {
      try {
        StringBuilder sb = new StringBuilder();
        checkerClasses.forEach(c->{
          Checker checker = allCheckers.get(c);
          if(checker!=null){
            try {
              checker.check(object);
            }catch (Exception e){
              sb.append(e.getMessage()).append(" ");
            }
          }
        });
        if(sb.length()>0) {
          constraintValidatorContext.disableDefaultConstraintViolation();
          constraintValidatorContext
              .buildConstraintViolationWithTemplate(sb.toString())
              .addConstraintViolation();
          return false;
        }else{
          return true;
        }
      }catch (Exception e){
        log.warn("", e);
        return false;
      }
    }
    
  }
}

应用服务

  @Transactional
  public Brand create(@Valid BrandCreateParam param){
    
    Brand brand = new Brand(
        idGen.generateId(),
        param.getName(),
        param.getLogo(),
        param.getScore(),
        param.getHidden()
    );
    
    brandRepo.create(brand);
    
    return brand;
  }

具体代码已经放在github上:Brand.java

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 199,636评论 5 468
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 83,890评论 2 376
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 146,680评论 0 330
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 53,766评论 1 271
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 62,665评论 5 359
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,045评论 1 276
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,515评论 3 390
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,182评论 0 254
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,334评论 1 294
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,274评论 2 317
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,319评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,002评论 3 315
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,599评论 3 303
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,675评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,917评论 1 255
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,309评论 2 345
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 41,885评论 2 341