cavis/python4j/python4j-core/src/test/java/PythonMultiThreadTest.java

169 lines
6.4 KiB
Java
Raw Normal View History

/*******************************************************************************
* Copyright (c) 2020 Konduit K.K.
*
* This program and the accompanying materials are made available under the
* terms of the Apache License, Version 2.0 which is available at
* https://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.
*
* SPDX-License-Identifier: Apache-2.0
******************************************************************************/
import org.nd4j.python4j.*;
import org.junit.Assert;
import org.junit.Test;
import javax.annotation.concurrent.NotThreadSafe;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import java.util.List;
@NotThreadSafe
public class PythonMultiThreadTest {
@Test
public void testMultiThreading1()throws Throwable{
final List<Throwable> exceptions = Collections.synchronizedList(new ArrayList<Throwable>());
Runnable runnable = new Runnable() {
@Override
public void run() {
try(PythonGIL gil = PythonGIL.lock()){
try(PythonGC gc = PythonGC.watch()){
List<PythonVariable> inputs = new ArrayList<>();
inputs.add(new PythonVariable<>("x", PythonTypes.STR, "Hello "));
inputs.add(new PythonVariable<>("y", PythonTypes.STR, "World"));
PythonVariable out = new PythonVariable<>("z", PythonTypes.STR);
String code = "z = x + y";
PythonExecutioner.exec(code, inputs, Collections.singletonList(out));
Assert.assertEquals("Hello World", out.getValue());
System.out.println(out.getValue() + " From thread " + Thread.currentThread().getId());
}
}catch (Throwable e){
exceptions.add(e);
}
}
};
int numThreads = 10;
Thread[] threads = new Thread[numThreads];
for (int i = 0; i < threads.length; i++){
threads[i] = new Thread(runnable);
}
for (int i = 0; i < threads.length; i++){
threads[i].start();
}
Thread.sleep(100);
for (int i = 0; i < threads.length; i++){
threads[i].join();
}
if (!exceptions.isEmpty()){
throw(exceptions.get(0));
}
}
@Test
public void testMultiThreading2()throws Throwable{
final List<Throwable> exceptions = Collections.synchronizedList(new ArrayList<Throwable>());
Runnable runnable = new Runnable() {
@Override
public void run() {
try(PythonGIL gil = PythonGIL.lock()){
try(PythonGC gc = PythonGC.watch()){
PythonContextManager.reset();
PythonContextManager.reset();
List<PythonVariable> inputs = new ArrayList<>();
inputs.add(new PythonVariable<>("a", PythonTypes.INT, 5));
String code = "b = '10'\nc = 20.0 + a";
List<PythonVariable> vars = PythonExecutioner.execAndReturnAllVariables(code, inputs);
Assert.assertEquals("a", vars.get(0).getName());
Assert.assertEquals(PythonTypes.INT, vars.get(0).getType());
Assert.assertEquals(5L, (long)vars.get(0).getValue());
Assert.assertEquals("b", vars.get(1).getName());
Assert.assertEquals(PythonTypes.STR, vars.get(1).getType());
Assert.assertEquals("10", vars.get(1).getValue().toString());
Assert.assertEquals("c", vars.get(2).getName());
Assert.assertEquals(PythonTypes.FLOAT, vars.get(2).getType());
Assert.assertEquals(25.0, (double)vars.get(2).getValue(), 1e-5);
}
}catch (Throwable e){
exceptions.add(e);
}
}
};
int numThreads = 10;
Thread[] threads = new Thread[numThreads];
for (int i = 0; i < threads.length; i++){
threads[i] = new Thread(runnable);
}
for (int i = 0; i < threads.length; i++){
threads[i].start();
}
Thread.sleep(100);
for (int i = 0; i < threads.length; i++){
threads[i].join();
}
if (!exceptions.isEmpty()){
throw(exceptions.get(0));
}
}
@Test
public void testMultiThreading3() throws Throwable{
PythonContextManager.deleteNonMainContexts();
String code = "c = a + b";
final PythonJob job = new PythonJob("job1", code, false);
final List<Throwable> exceptions = Collections.synchronizedList(new ArrayList<Throwable>());
class JobThread extends Thread{
private int a, b, c;
public JobThread(int a, int b, int c){
this.a = a;
this.b = b;
this.c = c;
}
@Override
public void run(){
try{
PythonVariable<Long> out = new PythonVariable<>("c", PythonTypes.INT);
job.exec(Arrays.<PythonVariable>asList(new PythonVariable<>("a", PythonTypes.INT, a),
new PythonVariable<>("b", PythonTypes.INT, b)),
Collections.<PythonVariable>singletonList(out));
Assert.assertEquals(c, out.getValue().intValue());
}catch (Exception e){
exceptions.add(e);
}
}
}
int numThreads = 10;
JobThread[] threads = new JobThread[numThreads];
for (int i=0; i < threads.length; i++){
threads[i] = new JobThread(i, i + 3, 2 * i +3);
}
for (int i = 0; i < threads.length; i++){
threads[i].start();
}
Thread.sleep(100);
for (int i = 0; i < threads.length; i++){
threads[i].join();
}
if (!exceptions.isEmpty()){
throw(exceptions.get(0));
}
}
}