抽取非基本验证到规则文件 - A2D规则引擎
基本验证与业务验证,基本验证就是始终保持不变的验证规则,可以通过如下硬编码实现:
public class Order
{
[Required]
[Range(typeof(decimal), "", "")]
public decimal Price { get; set; } [Required]
[StringLength()]
public string Customer { get; set; } [Required(AllowEmptyStrings=true)]
[StringLength()]
public string StoreID { get; set; }
}
然后在用如下代码validate, 把错误放到List中:
private bool ValidateBasicRule(Order order)
{
List<KeyValuePair<string, string>> errors = order.IsValid();
if (errors.Count > )
{
this.AddRange(errors);
return false;
} return true;
} public static class DataAnnotationHelper
{
public static List<KeyValuePair<string, string>> IsValid<T>(this T o)
{
List<KeyValuePair<string, string>> errors = new List<KeyValuePair<string, string>>(); var descriptor = GetTypeDescriptor(typeof(T)); foreach (PropertyDescriptor propertyDescriptor in descriptor.GetProperties())
{
foreach (var validationAttribute in propertyDescriptor.Attributes.OfType<ValidationAttribute>())
{
if (!validationAttribute.IsValid(propertyDescriptor.GetValue(o)))
{
errors.Add(new KeyValuePair<string, string>(propertyDescriptor.Name, validationAttribute.FormatErrorMessage(propertyDescriptor.Name)));
}
}
}
return errors;
}
private static ICustomTypeDescriptor GetTypeDescriptor(Type type)
{
return new AssociatedMetadataTypeTypeDescriptionProvider(type).GetTypeDescriptor(type);
}
}
然后说说业务规则的易变
SaaS程序,或者业务规则极其易变时,就要采用其他方法来做了,不可能每个公司都用设计模式分开写(虽然也行,但是不方便,公司业务规则多了后,对这些规则代码的管理就是很高的成本,而且要developer来负责)。所以要用规则文件来分开规则的编写,好处:
- 把修改的职责交给别人,比如项目经理、项目实施人员
- 代码不需要重新编译就能实现业务规则的修改
我们来解决下这个易变问题,假设有2公司:A和B。
A公司验证规则:
- 基本验证(就是Order类的验证规则的硬编码)
- 自定义验证规则:当前Order的下单网址必须来自于这几个url,如:www.cnblogs.com、www.cnblogs1.com、www.cnblogs2.com
B公司验证规则:
- 基本验证(同上)
- 自定义验证规则:无
如果用A2D规则引擎来解决的话,需要怎么做呢?
第一步当然是编写规则文件,A公司的规则文件:
declare
allowStores=["www.cnblogs.com", "www.cnblogs1.com", "www.cnblogs2.com"]
end declare
rule "test"
when
!_.contains(allowStores, entity.StoreID)
then
errors.Add("StoreID", "StoreID value not right")
end rule
由于B公司没有自定义规则,因此不需要编写相应的规则文件
第二步,调整验证逻辑,如下:
public class OrderService : BrokenRulesHolder
{
public int PlaceOrder(Order order)
{
this.ClearBrokenRules();
//进行基本规则验证
if (!ValidateBasicRule(order))
return -; //进行针对不同公司的规则验证
if (!ValidateCompanyRule(order))
return -; //其他操作,比如插入数据库 return ;
} private bool ValidateCompanyRule(Order order)
{
BrokenRulesHolder tempBrokenRulesHolder = new BrokenRulesHolder();
string rulePath = System.IO.Path.Combine(AppDomain.CurrentDomain.BaseDirectory, "Rules", "OrderValidations", SessionContext.CompanyID + ".r");
using (RuleEngine engine = new RuleEngine(false)) //false代表:如果规则文件不存在不会报错,而是忽略,默认为true
{
engine.BindRulePath(rulePath); //将规则文件的全路径传入引擎 engine.SetParameter("entity", order);
engine.SetParameter("errors", tempBrokenRulesHolder); engine.Process();
}
if (tempBrokenRulesHolder.BrokenRules.Count > )
{
this.AddRange(tempBrokenRulesHolder.BrokenRules);
return false;
}
return true;
} private bool ValidateBasicRule(Order order)
{
List<KeyValuePair<string, string>> errors = order.IsValid();
if (errors.Count > )
{
this.AddRange(errors);
return false;
} return true;
}
}
BrokenRule.cs代码:
public class BrokenRulesHolder
{
private List<KeyValuePair<string, string>> brokenRules = new List<KeyValuePair<string, string>>();
public List<KeyValuePair<string, string>> BrokenRules
{
get
{
return this.brokenRules.AsReadOnly().ToList();
}
}
public void Add(string key, string msg)
{
brokenRules.Add(new KeyValuePair<string, string>(key, msg));
}
public void AddRange(List<KeyValuePair<string, string>> rules)
{
brokenRules.AddRange(rules);
}
public void ClearBrokenRules()
{
brokenRules.Clear();
}
}
demo代码已经更新到A2D框架中了,这里就不upload运行图了。
A2D规则引擎已经内嵌了underscore1.5.2,因此规则定义文件(.r文件)中可以写相应的underscore函数来简化写法。
规则文件说明:
declare
allowStores1=["www.cnblogs.com", "www.cnblogs1.com", "www.cnblogs2.com"]
allowStores2=["www.cnblogs.com", "www.cnblogs1.com", "www.cnblogs2.com"]
end declare
rule "rule 1"
when
!allowStores1.contains(entity.StoreID)
then
errors.Add("StoreID", "StoreID value not right")
end rule
rule "rule 2"
when
!allowStores2.contains(entity.StoreID)
then
errors.Add("StoreID", "StoreID value not right")
end rule
declare section:
- 可以不写,但是最多只能存在1个这样的section
- 每行可以有逗号,也可以没有逗号
- 这个section的本意是:当前规则文件的变量全局定义、初始化工作
- 开头为declare,小写
- 结尾为end declare,小写
rule section:
- 必须存在1到多个
- 每行可以有逗号,也可以没有逗号
- 开头为rule "一些描述",小写
- 结尾为end rule,小写
- 如果存在3个rule时,最终会变成这样的js逻辑
加载underscore的js代码 加载declare section的js代码 if(rule1 conditions)
{
执行rule1的then语句
}
else if(rule2 conditions)
{
执行rule2的then语句
}
else if(rule3 conditions)
{
执行rule3的then语句
}
大家可以下载代码进行自己修改,比如可以插入自己编写的js代码来更加简化或者更加贴近项目的自定义方法、函数。
抽取非基本验证到规则文件 - A2D规则引擎的更多相关文章
- Drools 规则文件语法概述
概述(Overview) 以.drl为扩展名的文件,是Drools中的规则文件,规则文件的编写,遵循Drools规则语法.下面详细介绍一下Drools规则文件语法.具体参考官方文档: https:// ...
- [原创]java WEB学习笔记71:Struts2 学习之路-- struts2常见的内建验证程序及注意点,短路验证,非字段验证,错误消息的重用
本博客的目的:①总结自己的学习过程,相当于学习笔记 ②将自己的经验分享给大家,相互学习,互相交流,不可商用 内容难免出现问题,欢迎指正,交流,探讨,可以留言,也可以通过以下方式联系. 本人互联网技术爱 ...
- Struts2的输入校验(1)——校验规则文件的编写
Struts2的输入校验(1) --校验规则文件的编写 Struts2提供了基于验证框架的输入校验,所有的输入校验只要编写配置文件,Struts2的验证框架将会负责进行服务器校验和客户端校验. 注: ...
- A2D规则引擎
A2D规则引擎 写了个简单的规则引擎,普通情况够用了: 比如2家公司有各自的利率计算规则,如下: 在C#方面,没有写在C#的业务逻辑代码中,而是移到了外部规则文件中,如(ACompanyRatePol ...
- Struts(二十四):短路验证&重写实现转换验证失败时短路&非字段验证
短路验证: 若对一个字段使用多个验证器,默认情况下会执行所有的验证.若希望前面的验证器没有通过,后面的验证器就不再执行,可以使用短路验证. 1.如下拦截器,如果输入字符串,提交表单后,默认是会出现三个 ...
- 如何在jenkins的maven项目中,用mvn命令行指定findbugs的黑名单规则文件
一:问题背景 最近在研究jenkins的过程中,针对maven项目,打算添加findbugs进行静态检查,但我不太想在项目的pom中进行修改,最好可以只修改jenkins的job配置,即配置外部化. ...
- 极其好用好学的规则引擎 - A2D规则引擎
写了个简单的规则引擎,普通情况够用了: 比如2家公司有各自的利率计算规则,如下: 在C#方面,没有写在C#的业务逻辑代码中,而是移到了外部规则文件中,如(ACompanyRatePolicy.r): ...
- AWVS11提取规则文件
在这里给大家分享一个获取AWVS规则文件的思路. 目前我提取的是17年4月份的扫描规则. 后面如果规则更新,可以自行提取 官网: https://www.acunetix.com/vulne ...
- .NET开源工作流RoadFlow-Bug修改-1.8.2表单验证时ueditor编辑非空验证无效
RoadFlow生成的表单,Ueditor编辑器不能进行非空验证的BUG修改: 1.修改控制器:WorkFlowFormDesignerController红框处: 2.修改js文件:Scripts/ ...
随机推荐
- ie6常见的兼容性问题
1.<!DOCTYPE HTML>文档类型的声明. 产生条件:IE6浏览器,当我们没有书写这个文档声明的时候,会触发IE6浏览器的怪异解析现象: 解决办法:书写文档声明. 2.不同浏览器当 ...
- [Objective-C] Copy 和 MutableCopy
看了几篇文章,因为文章很新手向,所以内容很繁琐.故整理一下重点,写了测试程序去了解几个知识点,不讨论基本概念.新博客wossoneri.com传送门 非集合类对象的copy与mutableCopy / ...
- 使用Visual Studio Team Services敏捷规划和项目组合管理(一)——使用团队、区域和迭代
使用Visual Studio Team Services敏捷规划和项目组合管理(一)--使用团队.区域和迭代 概述 在本实验,你将会了解Visual Studio Team Services提供的敏 ...
- Oracle笔记----oracle数字类型number自增
创建序列 create sequence seq_student start increment maxvalue nominvalue nocycle nocache; 创建触发器 create o ...
- solidity高级理论(三):时间单位与view
solidity高级理论(三):时间单位与view 关键字:时间单位.view.Gas优化 solidity使用自己的本地时间单位 变量 now 将返回当前的unix时间戳(自1970年1月1日以来经 ...
- Linux用户抢占和内核抢占详解(概念, 实现和触发时机)--Linux进程的管理与调度(二十)
1 非抢占式和可抢占式内核 为了简化问题,我使用嵌入式实时系统uC/OS作为例子 首先要指出的是,uC/OS只有内核态,没有用户态,这和Linux不一样 多任务系统中, 内核负责管理各个任务, 或者说 ...
- MySql基本使用方法
一,基本概念 1, 常用的两种引擎: (1) InnoDB a,支持ACID,简单地说就是支持事务完整性.一致性: b,支持行锁,以及类似ORACLE的一 ...
- February 16th, 2018 Week 7th Friday
Full of luck, health and cheer. We wish you a Happy Chinese New Year! 春节快乐,万事如意! From Shanbay. Today ...
- wangedit
<template> <el-row id="AddRoom"> <el-col :span="5">.</el-co ...
- Zookeeper源码编译为Eclipse工程(win7下Ant编译)
前言 ZooKeeper是雅虎的.用Ant进行软件构建. 千里之行,始于足下.想看源码的第一步,是下载源码并导入某个IDE工具. Ant http://ant.apache.org/ Windows: ...