我想在 Drools 7.21 中评估 DMN 1.2 中新增的 FEEL-功能,如 sqrt() 或 modulo(),但该方法
dmnRuntime.evaluateAll(dmnModel,context)
全部返回状态为"成功"的值"null"(仅适用于新函数)。我做错了什么或缺少了什么?
DMN 文件如下所示:
<?xml version="1.0" encoding="UTF-8" standalone="no"?>
<semantic:definitions xmlns:semantic="http://www.omg.org/spec/DMN/20180521/MODEL/" xmlns="http://www.trisotech.com/definitions/_56fd6445-ff6a-4c28-8206-71fce7f80436" xmlns:feel="http://www.omg.org/spec/FEEL/20140401" xmlns:xsd="http://www.w3.org/2001/XMLSchema" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" exporter="DMN Modeler" exporterVersion="6.0.1" id="_56fd6445-ff6a-4c28-8206-71fce7f80436" name="Sqrt-Function" namespace="http://www.trisotech.com/definitions/_56fd6445-ff6a-4c28-8206-71fce7f80436" >
<semantic:decision id="_cf6124bd-9907-4ac0-b4fd-59a962dbc502" name="square_root">
<semantic:variable id="_edaf978e-3634-4e52-8244-5fd4e16fd257" name="square_root" typeRef="feel:number"/>
<semantic:literalExpression id="_c990c3b2-e322-4ef9-931d-79bcdac99686">
<semantic:text>sqrt(81)</semantic:text>
</semantic:literalExpression>
</semantic:decision>
</semantic:definitions>
在"dmnModel"中导入文件后:
DMNMarshaller marshaller = new org.kie.dmn.backend.marshalling.v1x.XStreamMarshaller();
FileInputStream fis = new FileInputStream( dmnFile );
Definitions unmarshal = marshaller.unmarshal( new InputStreamReader( fis ) );
DMNCompiler compiler = DMNFactory.newCompiler();
DMNModel dmnModel = compiler.compile(unmarshal);
我这样称呼流口水评估:
KieContainer kieContainer = KieHelper.getKieContainer(ks.newReleaseId("org.kie", "dmn-test-"+UUID.randomUUID(), "1.2"));
DMNRuntime dmnRuntime = kieContainer.newKieSession().getKieRuntime(DMNRuntime.class);
((DMNRuntimeImpl) dmnRuntime).setOption(new RuntimeTypeCheckOption(true));
DMNResult result = dmnRuntime.evaluateAll(dmnModel, context);
最好不要(取消)马歇尔,也不必手动编译DMN文件;而是使用来自KJAR的KieContainer标准构建的标准方法;如文档中的用户指南所述。
换句话说,这适用于您的 DMN 文件:
KieServices kieServices = KieServices.Factory.get();
KieContainer kieContainer = kieServices.getKieClasspathContainer();
DMNRuntime dmnRuntime = KieRuntimeFactory.of(kieContainer.getKieBase()).get(DMNRuntime.class);
DMNModel dmnModel = dmnRuntime.getModel(namespace, modelName);
DMNContext context = dmnRuntime.newContext();
((DMNRuntimeImpl) dmnRuntime).setOption(new RuntimeTypeCheckOption(true));
DMNResult result = dmnRuntime.evaluateAll(dmnModel, context);
因此产生9
。
如果你真的想使用 KieHelper,最好将 DMN 文件作为 KieResource,传递给KieHelper.getKieContainer(...)
的调用,例如:
KieContainer kieContainer = KieHelper.getKieContainer(ks.newReleaseId("org.kie", "dmn-test-" + UUID.randomUUID(), "1.2"),
ks.getResources().newFileSystemResource(new File(dmnFile)));
DMNRuntime dmnRuntime = KieRuntimeFactory.of(kieContainer.getKieBase()).get(DMNRuntime.class);
((DMNRuntimeImpl) dmnRuntime).setOption(new RuntimeTypeCheckOption(true));
DMNModel dmnModel = dmnRuntime.getModel(namespace, modelName);
DMNContext context = dmnRuntime.newContext();
DMNResult result = dmnRuntime.evaluateAll(dmnModel, context);
System.out.println(result);
您可以根据需要将调用ks.getResources().newFileSystemResource(...)
更改为基于 URL、类路径、字节、... 的资源,具体取决于您的用例。这样,KieHelper将负责解组
此外,第二个片段也适用于您的 DMN 文件,因此产生了9
。
代码中的问题是DMNCompiler的初始化实际上并不是由用户手动调用的,事实上,文档中没有任何地方需要手动管理它;上述两种方法都会将其委托给KieContainer/KieHelper的内部,这将是标准方式。
我建议按照文档中所述的 KieContainer 构建,如前一个例子中详细介绍的那样,但我希望这个答案可以帮助您解决任何一种方式 - 本地两者都为我工作。
希望这有帮助!