From cf1162dfc617a304d1a2e0d5c1ccf7ba5b8ecbd5 Mon Sep 17 00:00:00 2001 From: Toshiya Kobayashi Date: Thu, 19 May 2022 16:04:47 +0900 Subject: [PATCH 1/2] [DROOLS-6961] NullPointerException in LambdaConsequence with global in executable-model --- .../consequence/LambdaConsequence.java | 4 +- .../concurrency/GlobalConcurrencyTest.java | 142 ++++++++++++++++++ 2 files changed, 145 insertions(+), 1 deletion(-) create mode 100644 drools-test-coverage/test-compiler-integration/src/test/java/org/drools/compiler/integrationtests/concurrency/GlobalConcurrencyTest.java diff --git a/drools-model/drools-model-compiler/src/main/java/org/drools/modelcompiler/consequence/LambdaConsequence.java b/drools-model/drools-model-compiler/src/main/java/org/drools/modelcompiler/consequence/LambdaConsequence.java index 5a87e33a148..1511a922281 100644 --- a/drools-model/drools-model-compiler/src/main/java/org/drools/modelcompiler/consequence/LambdaConsequence.java +++ b/drools-model/drools-model-compiler/src/main/java/org/drools/modelcompiler/consequence/LambdaConsequence.java @@ -45,6 +45,7 @@ public class LambdaConsequence implements Consequence { private TupleFactSupplier[] factSuppliers; private GlobalSupplier[] globalSuppliers; private Object[] facts; + private boolean suppliersInitialized = false; private FactHandleLookup fhLookup; @@ -117,7 +118,7 @@ private static InternalFactHandle getOriginalFactHandle( InternalFactHandle hand } private Object[] fetchFacts( KnowledgeHelper knowledgeHelper, InternalWorkingMemory workingMemory ) { - if (factSuppliers == null) { + if (!suppliersInitialized) { return initConsequence(knowledgeHelper, workingMemory); } @@ -218,6 +219,7 @@ private Object[] initConsequence( KnowledgeHelper knowledgeHelper, InternalWorki this.facts = facts; this.fhLookup = fhLookup; } + suppliersInitialized = true; return facts; } diff --git a/drools-test-coverage/test-compiler-integration/src/test/java/org/drools/compiler/integrationtests/concurrency/GlobalConcurrencyTest.java b/drools-test-coverage/test-compiler-integration/src/test/java/org/drools/compiler/integrationtests/concurrency/GlobalConcurrencyTest.java new file mode 100644 index 00000000000..1e87dfe7b1b --- /dev/null +++ b/drools-test-coverage/test-compiler-integration/src/test/java/org/drools/compiler/integrationtests/concurrency/GlobalConcurrencyTest.java @@ -0,0 +1,142 @@ +/* + * Copyright 2022 Red Hat, Inc. and/or its affiliates. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.drools.compiler.integrationtests.concurrency; + +import java.util.ArrayList; +import java.util.Collection; +import java.util.List; +import java.util.concurrent.CountDownLatch; +import java.util.concurrent.ExecutorService; +import java.util.concurrent.Executors; +import java.util.concurrent.TimeUnit; + +import org.drools.testcoverage.common.model.Person; +import org.drools.testcoverage.common.model.Result; +import org.drools.testcoverage.common.util.KieBaseTestConfiguration; +import org.drools.testcoverage.common.util.KieBaseUtil; +import org.junit.Test; +import org.junit.experimental.categories.Category; +import org.junit.runner.RunWith; +import org.junit.runners.Parameterized; +import org.kie.api.KieBase; +import org.kie.api.runtime.KieSession; +import org.kie.test.testcategory.TurtleTestCategory; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import static org.assertj.core.api.Assertions.assertThat; + +@RunWith(Parameterized.class) +@Category(TurtleTestCategory.class) +public class GlobalConcurrencyTest { + + private static final Logger LOGGER = LoggerFactory.getLogger(GlobalConcurrencyTest.class); + + protected static int LOOP = 3000; + protected static int MAX_THREAD = 30; + protected final KieBaseTestConfiguration kieBaseTestConfiguration; + + public GlobalConcurrencyTest(final KieBaseTestConfiguration kieBaseTestConfiguration) { + this.kieBaseTestConfiguration = kieBaseTestConfiguration; + } + + @Parameterized.Parameters(name = "KieBase type={0}") + public static Collection getParameters() { + Collection parameters = new ArrayList<>(); + parameters.add(new Object[]{KieBaseTestConfiguration.CLOUD_IDENTITY_MODEL_PATTERN}); // DROOLS-6961 : exec-model only + return parameters; + } + + @Test + public void testGlobalConcurrency() { + String str = + "package org.mypkg;" + + "import " + Person.class.getCanonicalName() + ";" + + "import " + Result.class.getCanonicalName() + ";" + + "global Result globalResult;" + + "rule R1 when\n" + + " $p1 : Person(name == \"Mark\")\n" + + "then\n" + + " globalResult.setValue($p1.getName() + \" is \" + $p1.getAge());\n" + + "end\n" + + "rule R2 when\n" + + " $p1 : Person(name == \"Edson\")\n" + + "then\n" + + " globalResult.setValue($p1.getName() + \" is \" + $p1.getAge());\n" + + "end"; + + List exceptionList = new ArrayList<>(); + + for (int i = 0; i < LOOP; i++) { + if (i % 100 == 0) { + System.out.println("loop : " + i); + } + + KieBase kieBase = KieBaseUtil.getKieBaseFromKieModuleFromDrl("global-test", kieBaseTestConfiguration, str); + + ExecutorService executor = Executors.newFixedThreadPool(MAX_THREAD); + final CountDownLatch latch = new CountDownLatch(MAX_THREAD); + for (int n = 0; n < MAX_THREAD; n++) { + executor.execute(new Runnable() { + + public void run() { + + KieSession ksession = kieBase.newKieSession(); + Result result = new Result(); + ksession.setGlobal("globalResult", result); + + ksession.insert(new Person("Mark", 37)); + ksession.insert(new Person("Edson", 35)); + ksession.insert(new Person("Mario", 40)); + + latch.countDown(); + try { + latch.await(); + } catch (InterruptedException e) { + LOGGER.error(e.getMessage(), e); + } + + try { + ksession.fireAllRules(); + } catch (Exception e) { + exceptionList.add(e); + } + + ksession.dispose(); + } + }); + } + + executor.shutdown(); + try { + executor.awaitTermination(100, TimeUnit.SECONDS); + } catch (InterruptedException e) { + LOGGER.error(e.getMessage(), e); + } + + if (!exceptionList.isEmpty()) { + break; + } + } + + if (exceptionList.size() > 0) { + LOGGER.error(exceptionList.get(0).getMessage(), exceptionList.get(0)); + } + + assertThat(exceptionList).isEmpty(); + + } +} From e9f887299fb363c9e7ca2645093d35fd72adf010 Mon Sep 17 00:00:00 2001 From: Toshiya Kobayashi Date: Thu, 19 May 2022 16:56:08 +0900 Subject: [PATCH 2/2] change Suppliers order instead of boolean flag --- .../modelcompiler/consequence/LambdaConsequence.java | 7 +++---- 1 file changed, 3 insertions(+), 4 deletions(-) diff --git a/drools-model/drools-model-compiler/src/main/java/org/drools/modelcompiler/consequence/LambdaConsequence.java b/drools-model/drools-model-compiler/src/main/java/org/drools/modelcompiler/consequence/LambdaConsequence.java index 1511a922281..44ed4829ead 100644 --- a/drools-model/drools-model-compiler/src/main/java/org/drools/modelcompiler/consequence/LambdaConsequence.java +++ b/drools-model/drools-model-compiler/src/main/java/org/drools/modelcompiler/consequence/LambdaConsequence.java @@ -45,7 +45,6 @@ public class LambdaConsequence implements Consequence { private TupleFactSupplier[] factSuppliers; private GlobalSupplier[] globalSuppliers; private Object[] facts; - private boolean suppliersInitialized = false; private FactHandleLookup fhLookup; @@ -118,7 +117,7 @@ private static InternalFactHandle getOriginalFactHandle( InternalFactHandle hand } private Object[] fetchFacts( KnowledgeHelper knowledgeHelper, InternalWorkingMemory workingMemory ) { - if (!suppliersInitialized) { + if (factSuppliers == null) { return initConsequence(knowledgeHelper, workingMemory); } @@ -212,14 +211,14 @@ private Object[] initConsequence( KnowledgeHelper knowledgeHelper, InternalWorki tupleFactSupplier.resolveAndStore(facts, workingMemory, current.getFactHandle(), fhLookup); } - this.factSuppliers = factSuppliers.toArray( new TupleFactSupplier[factSuppliers.size()] ); + // factSuppliers has to be last because factSuppliers is used as an initialization flag in fetchFacts(). See DROOLS-6961 this.globalSuppliers = globalSuppliers.isEmpty() ? null : globalSuppliers.toArray( new GlobalSupplier[globalSuppliers.size()] ); + this.factSuppliers = factSuppliers.toArray( new TupleFactSupplier[factSuppliers.size()] ); if (!workingMemory.getSessionConfiguration().isThreadSafe()) { this.facts = facts; this.fhLookup = fhLookup; } - suppliersInitialized = true; return facts; }