RuleEngine规则引擎底层改造AviatorScript 之公式规则

前情提要,看上一个文章,具体要实现的效果就是

当然上来的问题就是前端的问题,这个框首先他们用的是富文本,富文本传到后台的结果是前端脚本,带着h5的标签,后面改成了这个,当时这个东西其实和后端关系不大,就是如何处理换行符而已,使用这个之后换行符变成了/n

其实我们也是比着普通规则新建的一个功能,不过有很多的自己的设计

他们的实现是这样的

java 复制代码
     */
    @Override
    public Object run(RunTestRequest runTestRequest) {
        log.info("模拟运行规则:{}", runTestRequest.getCode());
        RuleEngineGeneralRulePublish rulePublish = this.ruleEngineGeneralRulePublishManager.lambdaQuery()
                .eq(RuleEngineGeneralRulePublish::getStatus, runTestRequest.getStatus())
                .eq(RuleEngineGeneralRulePublish::getGeneralRuleCode, runTestRequest.getCode())
                .eq(RuleEngineGeneralRulePublish::getWorkspaceCode, runTestRequest.getWorkspaceCode())
                .one();
        if (rulePublish == null) {
            // 如果测试找不到,用线上  此场景出现在只有一个线上的时候
            rulePublish = this.ruleEngineGeneralRulePublishManager.lambdaQuery()
                    .eq(RuleEngineGeneralRulePublish::getStatus, DataStatus.PRD.getStatus())
                    .eq(RuleEngineGeneralRulePublish::getGeneralRuleCode, runTestRequest.getCode())
                    .eq(RuleEngineGeneralRulePublish::getWorkspaceCode, runTestRequest.getWorkspaceCode())
                    .one();
            if (rulePublish == null) {
                throw new ValidException("找不到可运行的规则数据:{},{},{}", runTestRequest.getWorkspaceCode(), runTestRequest.getCode(), runTestRequest.getStatus());
            }
        }
        Input input = new DefaultInput(runTestRequest.getParam());
        log.info("初始化规则引擎");
        RuleEngineConfiguration ruleEngineConfiguration = new RuleEngineConfiguration();
        Container.Body<GeneralRule> generalRuleContainer = ruleEngineConfiguration.getGeneralRuleContainer();
        GeneralRule rule = GeneralRule.buildRule(rulePublish.getData());
        generalRuleContainer.add(rule);
        GeneralRuleEngine engine = new GeneralRuleEngine(ruleEngineConfiguration);
        // 加载变量
        engine.getConfiguration().setEngineVariable(this.ruleEngineConfiguration.getEngineVariable());
        return engine.execute(input, runTestRequest.getWorkspaceCode(), runTestRequest.getCode());
    }

然后我们改成了这个样子

java 复制代码
    @Override
    public Object formularun(RunTestRequest runTestRequest) {
        QueryWrapper<RuleEngineFunction2> wrapper = new QueryWrapper<>();
        List<RuleEngineFunction2> list = ruleEngineFunction2Manager.list(wrapper);
        for (int i = 0; i < list.size(); i++) {
            RuleEngineFunction2 ruleEngineFunction2 = list.get(i);
            //获取设置对应的方法名
            String className = ruleEngineFunction2.getClassName();
            String functionName = ruleEngineFunction2.getFunctionName();
            if (applicationContext.containsBean(className)) {
                AviatorFunction abstractFunction = (AviatorFunction)applicationContext.getBean(className);
                AviatorEvaluator.addFunction(abstractFunction);
            }
        }

        String funcation = runTestRequest.getFuncation();
        funcation = removeHtmlTags(funcation);
        System.out.println(funcation);
        Expression compiledExp = AviatorEvaluator.compile(funcation);

        Map<String, Object> param = runTestRequest.getParam();

        String res = String.valueOf(compiledExp.execute(param));
        System.out.println(res);
        return null;
    }

    public  String removeHtmlTags(String input) {
        return input.replaceAll("<[^>]*>", "");
    }
}

实现方式,但是这个时候出现了问题,就是函数底层实现的时候出现了问题,然后我们将函数的实现修改了,这个代码就几乎没有什么变动了,这个时候有一个另外的需求。就是需要有一个函数nvlFunction,这个函数是用来干啥的呢,就是说处理入参的,出现了这个函数就是表示有入参,这个函数一共两个参数,第一个参数,是取自函数这个功能的字段,要求是函数功能的函数code值,用来表示入参,另一个参数就是当这个入参不输入的时候,将这个参数赋值给对应的函数。

听着不难,实现的话我也是觉得挺简单的

java 复制代码
@Function
public class NvlFunction extends AbstractFunction {
    @Override
    public String getName() {
        return "nvlFunction";
    }

    @Override
    public AviatorObject call(Map<String, Object> env, AviatorObject arg1, AviatorObject arg2) {
        String letter1 = FunctionUtils.getStringValue(arg1, env);
        String letter2 = FunctionUtils.getStringValue(arg2, env);
        System.out.println("letter1"+letter1);
        System.out.println("letter2"+letter2);
        return new AviatorString(letter2);
    }
}

初版,发现不大行,如果参数没有穿进去不大对,

java 复制代码
@Function
public class NvlFunction extends AbstractFunction {
    @Override
    public String getName() {
        return "nvlFunction";
    }

    @Override
    public AviatorObject call(Map<String, Object> env, AviatorObject arg1, AviatorObject arg2) {
        if (arg1 != null && !"".equals(arg1)) {
            return arg1;
        }
        return arg2;
    }
}

然后换成了这个,发现也是不大行,即使为空的话这个对象AviatorObject 也不会为null

java 复制代码
@Function
public class NvlFunction extends AbstractFunction {
    @Override
    public String getName() {
        return "nvlFunction";
    }

    @Override
    public AviatorObject call(Map<String, Object> env, AviatorObject arg1, AviatorObject arg2) {
        Object param1 = arg1.getValue(env);
        if (param1 == null || param1.toString().isEmpty()) {
            return arg2;
        } else {
            return arg1;
        }
    }
}

发现了只有这个value是null,当这个入参是空的时候,至此改造全部完成了,剩下的就是他原本框架的小bug的修改以及小功能的新增了。

相关推荐
浮游本尊29 分钟前
Java学习第25天 - Spring Cloud Alibaba微服务生态
java
Cg1362691597434 分钟前
Super的详解
java
毕设源码-朱学姐44 分钟前
【开题答辩全过程】以 便利店库存管理系统为例,包含答辩的问题和答案
java·eclipse
Nero181 小时前
代码随想录二刷第九天 | 232.用栈实现队列、225. 用队列实现栈、20. 有效的括号、1047. 删除字符串中的所有相邻重复项
java
泉城老铁1 小时前
tomcat 部署springboot,线程经常断开导致数据库连接池关闭,如何解决
java·spring boot·后端
Nero181 小时前
代码随想录二刷第八天 | 28. 实现 strStr()、459.重复的子字符串
java
Le1Yu1 小时前
配置管理的配置与拉取、热更新的配置、动态路由
java
Seven971 小时前
剑指offer-34、第⼀次出现的字符
java
摇滚侠1 小时前
Spring Boot 3零基础教程,properties文件中配置和类的属性绑定,笔记14
java·spring boot·笔记
星光一影2 小时前
HIS系统天花板,十大核心模块,门诊/住院/医保全流程打通,医院数字化转型首选
java·spring boot·后端·sql·elementui·html·scss