Drools的LHS中的异常捕获

Mas*_* T. 16 java drools

我有一个问题,其中一个drools规则的when子句抛出一个MethodNotFoundException.我正在寻找一种方法来确定它在运行时的规则,以便能够从要使用的规则列表中删除它.

规则示例

Rule "FooBar"
when
 $V1 : Foo (    )  AND
 $V2 : FooBar(    ) from $V1.getGoodMethod()  AND
 $V3 : FooBarBar( status == "FooBar" ) from $V2.getBadMethod()  
reply : FooFooBar()
then
reply.getList().add("FooBar");
end
Run Code Online (Sandbox Code Playgroud)

因此,FooBar上的getBadMethod不存在.我想要一种方法来告诉它是什么规则,并将其从要使用的规则列表中删除.

尝试和失败的解决方案:

我已经尝试扩展DefaultAgendaEventListener并覆盖beforeActivationFired方法,将要触发的规则添加到列表中.我希望列表中的最后一个是抛出错误的那个,但遗憾的是它并没有那么成功.

我现在尝试在我的所有规则之前添加"始终为真"的规则.它们记录了它之后的规则的名称.问题是当"WHEN"子句中存在异常时,不会记录任何内容.这就好像在发生上述异常时不会触发规则.

问题都在于动态drools生成代码.我想采取双管齐下的方法来修复生成代码,并捕获像本文中列出的异常.

旁注:我确实检查了构建器中的错误.我从下面的代码中没有收到任何错误.

KnowledgeBuilderErrors errors = builder.getErrors();

if (!errors.isEmpty()) {
    for (KnowledgeBuilderError error : errors) {
        ...
    }
}
Run Code Online (Sandbox Code Playgroud)

小智 6

根据我的理解,在执行fireAllRules()方法之前,应遵循以下步骤:

  1. 将规则添加到包/知识生成器
  2. 验证规则中没有错误
  3. 在工作记忆中注入规则

当然,没有第2步就可以使用fireRules,但是这种做法可能导致此问题中提到的问题.如果我是你,我会按照以下逻辑来解决你的问题:

步骤1:

private RuleBase initialiseDrools() throws IOException, DroolsParserException {
    PackageBuilder packageBuilder = readRules();
    return addRulesToWorkingMemory(packageBuilder);
}
Run Code Online (Sandbox Code Playgroud)

第2步:

private PackageBuilder readRules() throws DroolsParserException, IOException {
    PackageBuilder packageBuilder = new PackageBuilder();
    PackageBuilder intermPackageBuilder = null;

    listOfReader = dynamicRuleReader(); // Here goes your application code

    for(Reader reader : listOfReader){ 
        try{
            intermPackageBuilder = new PackageBuilder();
            intermPackageBuilder.addPackage(reader);
            assertNoRuleErrors(intermPackageBuilder); // This is the core step. 
            // Above line throws an exception, every time a rules fails. You can persist this exception for production debugging
            packageBuilder.addPackage(reader);
        }catch(DroolsParserException | IOException e){
            logger.error("Rules contain error, so skip adding them to the Package Builder");
        }
    }

    return packageBuilder;
}
Run Code Online (Sandbox Code Playgroud)

第3步:

public void shouldFireAllRules() throws IOException, DroolsParserException {
    RuleBase ruleBase = initialiseDrools();
    WorkingMemory workingMemory = ruleBase.newStatefulSession();
    int expectedNumberOfRulesFired = 2; // Idealy this number should be equal to the number of total rules injected in the Working memory

    int actualNumberOfRulesFired = workingMemory.fireAllRules();

    assertThat(actualNumberOfRulesFired, is(expectedNumberOfRulesFired));
}
Run Code Online (Sandbox Code Playgroud)

使用上述方法,您将不会执行有错误的规则,并且不会出现上述情况.但是,我仍然认为你应该更多地关注产生错误规则的代码片段,以及上述方法只是为了跟踪和持久存在这种情况.