我正在尝试使用持久性设置kmodule。我在内部使用guice,所以春天的例子对我没什么帮助。
我目前的设置如下:
这很好用,但我只是以编程方式完成此操作(设置环境的EntityManager和TransactionManager)。现在我希望在kmodule中进行此配置,以便模块本身可以定义是否要保留它们。
问题是,我不知道如何为此创建会话,或者如何配置模块。例如,我能找到的最简单的代码片段看起来像这样:
<kmodule
xmlns="http://jboss.org/kie/6.0.0/kmodule">
<kbase name="kbase1">
<ksession name="ksession1"/>
</kbase>
</kmodule>
此时会话的创建方式如下:
KieServices ks = KieServices.Factory.get();
KieContainer kContainer = ks.getKieClasspathContainer();
KieSession kSession = kContainer.newKieSession(name);
但是,如何在模块中配置持久性,或者如果不可能,则如何在编程设置中传递正确的会话名称。我所拥有的是:
// Guice inject Entitymanager and TransactionManager and setup the Environment variable env
KieSessionConfiguration ksconf = ks.newKieSessionConfiguration();
ksconf.setOption(ClockTypeOption.get("realtime"));
JPAKnowledgeService.newStatefulKnowledgeSession(kbase, ksconf, env);
我在想,也许有办法通过KieSessionConfiguraton将会话名称传递给JPAKnowledgeService类,但我似乎无法找到任何方法。
或者,在kmodule本身配置它会更酷。我可以使用弹簧方法,但我有点怀疑注射会像那样工作。据我所知,Spring中的后处理器会进行持久性注入。即使我提供,我也不认为guice可以做到这一点?
由于
答案 0 :(得分:1)
所以我调查了它并挖掘了很多drools代码来尝试解决这个问题。
所有KieContainer#newSession都会查询模型,然后向基地询问会话配置的新会话:
public KieSession newKieSession(String kSessionName, Environment environment, KieSessionConfiguration conf) {
KieSessionModelImpl kSessionModel = (KieSessionModelImpl) getKieSessionModel(kSessionName);
if ( kSessionModel == null ) {
log.error("Unknown KieSession name: " + kSessionName);
return null;
}
if (kSessionModel.getType() == KieSessionModel.KieSessionType.STATELESS) {
throw new RuntimeException("Trying to create a stateful KieSession from a stateless KieSessionModel: " + kSessionName);
}
KieBase kBase = getKieBase( kSessionModel.getKieBaseModel().getName() );
if ( kBase == null ) {
log.error("Unknown KieBase name: " + kSessionModel.getKieBaseModel().getName());
return null;
}
KieSession kSession = kBase.newKieSession( conf != null ? conf : getKnowledgeSessionConfiguration(kSessionModel), environment );
wireListnersAndWIHs(kSessionModel, kSession);
registerLoggers(kSessionModel, kSession);
kSessions.put(kSessionName, kSession);
return kSession;
}
与此同时,这正是StoreService实现的功能,还有一些先进的哈利波特代码可以使持久化。然而它忽略了会话的配置,我几乎会说这是一个bug ..(可能会在此之后提出一个)
KieContainer创建的配置无论如何只考虑2个选项:
private KieSessionConfiguration getKnowledgeSessionConfiguration(KieSessionModelImpl kSessionModel) {
KieSessionConfiguration ksConf = KnowledgeBaseFactory.newKnowledgeSessionConfiguration();
ksConf.setOption( kSessionModel.getClockType() );
ksConf.setOption( kSessionModel.getBeliefSystem() );
return ksConf;
}
这意味着,我有2个选项可以解决这个问题:
要么写我自己的KieContainerImpl做正确的事情(为什么要打扰......)或模拟会话配置。我决定做第二个。
public KieSession createJPASession(final String kieBaseId, Optional<String> clockType, Optional<String> beliefSystem) {
log.info(String.format("Creating JPA Session for kieBase %s, clockType %s, beliefSystem %s", kieBaseId, clockType, beliefSystem));
KieBase kieBase = kContainer.getKieBase(kieBaseId);
KieSessionConfiguration ksConf = ks.newKieSessionConfiguration();
// Set this thing up manually. Looking at the impl/docs these are the only two options available to set.
// The Storage service will remove those options from a default session for whatever reason, however we can set this manually
// This means we can use the base configuration and have things run in a normal way
if (clockType.isPresent()) {
ksConf.setOption(ClockTypeOption.get(clockType.get()));
}
if (beliefSystem.isPresent()) {
ksConf.setOption(BeliefSystemTypeOption.get(beliefSystem.get()));
}
KieSession jpaSession = ks.getStoreServices().newKieSession(kieBase, ksConf, env);
sessions.put(jpaSession.getIdentifier(), jpaSession);
return jpaSession;
}
最后一段代码片段将创建配置的会话,同时使其持久化。
或者,通过将KieContainer转换为KieContainerImpl,可以访问查询KieModule中的KieSession对象所需的公共方法。 (很多Kie&#39; s的人)这样可以使用相同的方法并访问XML配置。 我的方法是混合使用(使用xml config for KieBase / Module,同时以编程方式设置信念系统和时钟类型)。原因是我不想依赖内部实现,我也不想打扰实现我自己的KieContainer并将其连接起来。
我希望这可以帮助别人。如果其他人知道&#34;正确&#34;这样做的方法,请发布。