DSL(特定领域语言): 开发 和 业务 共识的语言. 方便业务表达需求, 方便开发理解业务
一个业务需求(一般程序或一个接口)可以抽象成为: 按一定业务逻辑规则处理数据,然后返回数据
- 一个人可以用成百上千个属性组成,由这些属性衍生出新的属性(例如,好人/坏人) 返回一个业务结果(0..多个属性值)
- 一般接口: 查询数据库或接口,对数据进行简单逻辑计算,最后返回一些属性
框架提供一种通用流程: 从多种数据源获取数据转换成属性, 并按业务规则 DSL 执行
框架可以执行成千上万的这种通用程序逻辑, 可以动态修改流程中的每个过程(数据源,属性,规则),即时生效
可用于风控规则, 电商价格规则, 其它业务规则
- 决策: 抽象一个业务需求(一般程序或一个接口)为一条决策,一条决策包含多个业务规则
- 规则: 框架用自创 DSL规则 语言表达业务逻辑
规则 {
规则名 = '芝麻分限制'
拒绝 { 芝麻分 < 620 }
}
- 指标/字段/变量/属性: 规则中的变量 芝麻分
- 收集器: 指标值来源于数据源配置: 接口,SQL,脚本3种类型
例子 http://39.104.28.131:9090/ test:test
- 由 规则名 和多个 拒绝, 通过, 人工, 操作 组成(按顺序依次执行)
规则 {
规则名 = '大小比较'
拒绝 { 年龄 > 40 || 年龄 < 25 }
}
规则 {
规则名 = '通过规则'
通过 { 年龄 == 30 }
}
规则 {
规则名 = '赋值规则'
操作 { 产品代码 = 'xx' }
}
规则 { // 电商价格规则
规则名 = '条件赋值规则'
操作 {
if (当前时间 > '2021-03-02 00:00:00') {
单价 = 10
}
}
}
规则 {
规则名 = '列表判断规则'
风险地 = ["成都", "巴中"]
拒绝 { 工作地 in 风险地 }
}
规则 {
规则名 = '包含规则'
拒绝 { "$姓名".contains("xx") }
}
- 由 策略名 和多个 操作, 条件, 规则, 评分卡, 决策 组成(按顺序依次执行)
- 条件 函数返回false, 则跳出, 继续执行下一个策略
策略 {
策略名 = '004244'
操作 { jj_代码 = '004244' }
条件 { jj_名 }
规则 {
拒绝 { true }
}
}
模型 : 是由多个评分条目组成
- 普通评分条目: 由 属性名, 值匹配范围, 分值 组成
- 例: ['年龄', 18..24, 40]
- 例: ['工作成市', ['成都', '巴中'], 99]
- 函数分条目: 一个求值函数 组成
- 例: [{ -> 100}]
- 变量分条目: 一个数值变量 组成
- 例: ['芝麻分']
- 动态分条目: 一个数值变量,及其值计算函数 组成
- 例: ['逾期次数', {次数 -> -((次数?:0) * 100)}]
- 模型分计算: 依次遍历每个评分条目, 得到匹配的分值相加
- 最终评分 = 基础分 + 模型计算得分
赋值变量 : 把评分结果赋值给的变量
策略 {
策略名 = '测试评分卡'
评分卡 {
评分卡名 = "测试评分卡"
基础分 = 600
模型 = [
['年龄', 18..24, 40],
['年龄', 25..45, 80],
['年龄', 46..60, 50],
['性别', 'F', 50],
['性别', 'M', 60],
[{ -> 100}], // 函数分
['芝麻分'], //变量值分
['逾期次数', {次数 -> -((次数?:0) * 100)}], // 动态分
['工作成市', ['成都', '巴中'], 99]
]
赋值变量 = '评分结果'
}
规则 {
规则名 = "评分判断"
拒绝 { 评分结果 < 60 }
}
}
- 嵌套其它决策执行(注意不要循环嵌套)
- 会形成一条单独的决策执行记录(id为父决策的id为前缀)
- 异步:
- 不配置默认为false; 为true时, 输出 配置不生效
- 值为false: 并且 当子决策执行结果为Reject的时,会结束当前决策执行
- 传参: 共享当前决策数据给子决策使用
- 输出: 会设置 到当前执行上下文, 可以使用. 没有配置时: 则使用子决策的返回属性作为 输出
策略 {
策略名 = '测试子决策'
决策 {
决策id = "cd7049e7e76441fa8315c2e1bd883042"
// 不配置默认为false, 异步为true时, 输出则不生效
// 异步 = true
// 左(子决策) = 右(父决策)
传参 {
p1 = 1
p2 = "aa"
p3 = p3 // p3的值从父决策来
}
// 左(父决策) = 右(子决策)
输出 {
p111 = p1 // p1的值从子决策来
p222 = p3 // p3的值从子决策来
}
}
}
- 由 决策名, 决策描述 和多个 操作, 策略 组成(按顺序依次执行)
- 触发当前决策: http://ip:port/decision/{id}/decide
- 接口返回: 3种结果(Accept, Reject, Review) 和多个返回属性值(由配置指定)
// 返回示例
{
"decideId": "647f297a2e4540dfa93991b5b6e7b44d",
"result": "Accept",
"decisionId": "jj_analyse",
"status": "0000",
"desc": null,
"data": {
"jj_code": "165525"
}
}
- 依次从上往下执行
决策名 = '测试1'
决策描述 = ''
// 返回的调用决策方的结果属性值
// 返回属性 '身份证号码'
// 预操作执行
操作 {}
策略 {
策略名 = '预处理'
// 条件 { true }
规则 {
规则名 = '年龄限制'
拒绝 { 年龄 > 40 || 年龄 < 25 }
}
}
决策名 = '测试1'
策略 {
策略名 = '测试策略'
规则 {
规则名 = '使用函数'
操作 {
钉钉消息("发个消息")
}
}
}
函数定义("钉钉消息") {String msg ->
Utils.http()
.post("https://oapi.dingtalk.com/robot/send?access_token=7e9d8d97e6b5e76a6a07b0c5d7c31e82f0fbdb8ced1ac23168f9fd5c28c57f1a")
.jsonBody(JSON.toJSONString(
[msgtype: 'text', text: [content: "Fund: " + msg], at: ['isAtAll': false]]
)).debug().execute()
}
拒绝 { 年龄 > 50 }
支持从SQL(数据库),http(接口),script(自定义groovy脚本函数)获取数据
静态权限
- 权限管理, 登陆, 用户-创建, 决策-创建, 字段-添加, 字段-查看, 字段-更新, 字段-删除, 收集器-添加, 收集器-查看, 收集器-更新, 收集器-删除, 操作历史查看
权限管理: 可修改任何用户的任何权限
用户-创建: 创建者默认拥有被创建的用户的所有动态权限
操作历史查看: 目前支持 决策-查看-{某个决策}, 字段-查看, 收集器-查看
动态资源(用户)权限
- 被创建用户默认拥有 用户-查看-{自己} 此动态权限
- 每添加一个用户 就会生成与之对应的以下动态权限
用户-查看-{用户1}: 能搜索到此用户1
用户-删除-{用户1}: 能删除用户1
用户-重置密码-{用户1}: 能为用户1重置密码
用户-继承决策权限-{用户1}: 能自动拥有用户1新创建的决策的所动态有权限
用户-修改权限-{用户1}: 能修改指定用户1权限
- 如果: 修改用户有 权限管理, 则可修改用户1的所有权限
- 否则: 能修改用户1的权限只能是, 当前用户所拥有的权限(除 用户-创建)
动态资源(决策)权限
- 创建者默认自动拥有生成动态权限
- 每添加一个决策 就会生成与之对应的以下动态权限
决策-查看-{决策1}: 拥有查看决策1的权限
决策-删除-{决策1}: 拥有删除决策1的权限
决策-更新-{决策1}: 拥有更新决策1的权限
- 超级用户(系统管理员): 拥有 权限管理 的用户
- 用户管理员: 拥有 用户-创建 的用户
- 普通用户: 非 权限管理 和 用户-创建 的用户
系统基于java轻量级框架: Tiny
- git clone https://gitee.com/xnat/grule.git 或 git clone https://github.com/xnat9/grule.git
- linux: sh start.sh -Xmx512m -Xms512m -Dlog.totalSizeCap=50G
- windows: ./start -Xmx512m -Xms512m
- api文档地址: http://localhost:9090/api-doc/
jdk8, gradle7+
数据库要求: mysql5.7+(最好是8.0.4+), MariaDB10.2+
-- 创建数据库
CREATE DATABASE IF NOT EXISTS rule DEFAULT CHARSET utf8 COLLATE utf8_general_ci;
# 替换 conf/app.properties 中的 jpa_rule.url 配置项
jpa_rule.url=jdbc:mysql://localhost:3306/rule?useSSL=false&user=root&password=root&useUnicode=true&characterEncoding=utf-8&allowPublicKeyRetrieval=true
需要添加Groovy sdk 3.0.8
IntelliJ IDEA 运行 main.groovy 启动
- feat: 容器镜像 Dockerfile
- feat: 重新执行决策(自动/手动)
- opt: 实体关联(无外键)
- opt: 深入优化决策DSL变量作用域
- feat: 新增 DSL SPEC: 接口
- feat: redis 收集器
- upgrade: groovy 4.0.1
- feat: 字符串 in 或者 contains 操作符