如何在运行时构建kmodule.xml以更新drl文件中所做的更改

时间:2017-09-25 06:40:31

标签: java spring-boot drools kie

我是流口水的新手。我正面临着drools引擎的问题。我有一个用户可以从Web界面创建规则,我使用一些代码将创建这些规则中的drl文件。直到这里一切都按预期工作。现在,用户将通过post请求向暴露的api提供输入,并根据一些参数值,我们将选择一个drl文件并执行规则。早些时候,我正在做的每一个请求我都在读取我生成的kieFileSystem并构建它的drl文件。因此,如果用户发送3个参数,那么我逐个构建3个不同的文件并执行它们很慢。(3个参数需要大约1秒,参数没有限制所以他也可以发送10-12个很慢。)

public <T> void getScore(T droolsInput, String filePath, String fileName) throws Exception {
        KieServices kieServices = KieServices.Factory.get();
        KieFileSystem kfs = kieServices.newKieFileSystem();
        FileInputStream fis = new FileInputStream(filePath);
        kfs.write(fileName, kieServices.getResources().newInputStreamResource(fis));
        KieBuilder kieBuilder = kieServices.newKieBuilder(kfs).buildAll();
        if (kieBuilder.getResults().hasMessages(Level.ERROR)) {
            throw new RuntimeException("Build Errors:\n" + kieBuilder.getResults().toString());
        }
        KieContainer kieContainer = kieServices.newKieContainer(kieServices.getRepository().getDefaultReleaseId());
        KieSession kieSession = kieContainer.newKieSession();
        kieSession.insert(droolsInput);
        kieSession.fireAllRules();
        kieSession.dispose();
        fis.close();
    }

这是我之前使用的代码。现在我开始了解kmodule.xml,我如何在启动应用程序时构建kbases和ksession,我的响应时间从1秒到近300毫秒。但问题是我需要在META-INF文件夹中启动我的应用程序之前创建kmodule.xml文件,否则它无法正常工作。(我不知道我做错了什么或者是什么)。因此,当用户使用Web界面更新规则时,我们再次构建drl文件,当他点击api时,我们希望根据新规则给他输出。我们不想重新启动应用程序来重新配置规则。他们可以使用该服务的许多用户。所以任何人都可以帮助我。基本上我想要的是保持规则更新,并使响应时间尽可能低。

这是我用来生成kmodule.xml的代码,这对我不起作用。

private Resource[] getRuleFiles() throws IOException {
        ResourcePatternResolver resourcePatternResolver = new PathMatchingResourcePatternResolver();
        return resourcePatternResolver.getResources("classpath*:" + RULES_PATH + "**/*.*");
    }

    @Bean
    public KieContainer buildKIEContainer() throws IOException {
        final KieRepository kieRepository = getKieServices().getRepository();

        kieRepository.addKieModule(new KieModule() {
            @Override
            public ReleaseId getReleaseId() {
                return kieRepository.getDefaultReleaseId();
            }
        });
        System.out.println("#############################################################################");
        System.out.println(getRuleFiles()[0].getURL());
        // System.out.println();
        KieModuleModel kieModuleModel = getKieServices().newKieModuleModel();
        // KieFileSystem kieFileSystem = kieFileSystem();
        for (Resource file : getRuleFiles()) {
            String fileNameWithoutExt = file.getFilename().split("\\.")[0];
            KieBaseModel kieBaseModel1 = kieModuleModel.newKieBaseModel(fileNameWithoutExt).setDefault(true)
                    .setEqualsBehavior(EqualityBehaviorOption.EQUALITY)
                    .setEventProcessingMode(EventProcessingOption.STREAM);
            kieBaseModel1.newKieSessionModel("session_" + fileNameWithoutExt).setDefault(true)
                    .setType(KieSessionModel.KieSessionType.STATEFUL).setClockType(ClockTypeOption.get("realtime"));
            kieBaseModel1.addPackage("rules." + fileNameWithoutExt);
            FileInputStream fis = new FileInputStream(file.getFile());
            kieFileSystem.write("src/main/resources/rules/" + fileNameWithoutExt + "/" + file.getFilename(),
                    getKieServices().getResources().newInputStreamResource(fis));

        }
        System.out.println(kieModuleModel.toXML());
        kieFileSystem.writeKModuleXML(kieModuleModel.toXML());
        KieBuilder kieBuilder = getKieServices().newKieBuilder(kieFileSystem);
        kieBuilder.buildAll();
        if (kieBuilder.getResults().hasMessages(Level.ERROR)) {
            throw new RuntimeException("Build Errors:\n" + kieBuilder.getResults().toString());
        }

        return getKieServices().newKieClasspathContainer("mycontainer");

    }

    public KieServices getKieServices() {
        return KieServices.Factory.get();
    }

    @Bean
    public KieFileSystem kieFileSystem() throws IOException {
        return getKieServices().newKieFileSystem();

    }

这是我用来触发规则的代码。

@Autowired
    private KieContainer kieContainer;

    public <T> void getScore(T droolsInput, String filePath, String fileNameWithPath, String filename)
            throws Exception {
        KieSession kieSession = kieContainer.newKieSession("session_" + filename.split("\\.")[0]);
        kieSession.insert(droolsInput);
        kieSession.fireAllRules();
        kieSession.dispose();
    }

所以基本上上面的fireRules代码在项目的META-INF文件夹中存在kmodule.xml时有效,否则它不起作用。

1 个答案:

答案 0 :(得分:0)

应该可以单独编译每个DRL文件以获得KiePackage,并将它们任意组合(根据参数)到生成会话的KieBase中。但我认为这需要访问内部API,如有更改,恕不另行通知。

因此我会使用另一种方法。

添加

class Parameters {
    Set<String> set = ...
}

和每个X.drl的一条规则:

rule selectX
when
    Parameters( set contains "X" ) # pattern to select X
then
end

并在X.drl中编写所有规则

rule some_rule_in_X
extends selectX
    // ...

或者,编写模式以在每个规则中选择X.

要激活,请插入从post请求派生的类Parameters对象。