From d84735042227381f5cdde9f70a58dec7cb071d6c Mon Sep 17 00:00:00 2001 From: Phil Elson Date: Tue, 2 May 2023 17:14:22 +0200 Subject: [PATCH 1/2] Handle None being passed explicitly to startJVM --- jpype/_core.py | 2 +- test/jpypetest/test_startup.py | 14 ++++++++++---- 2 files changed, 11 insertions(+), 5 deletions(-) diff --git a/jpype/_core.py b/jpype/_core.py index bdc4a0e3c..07f246c5d 100644 --- a/jpype/_core.py +++ b/jpype/_core.py @@ -211,7 +211,7 @@ def startJVM( # JVM path if jvmargs: # jvm is the first argument the first argument is a path or None - if jvmargs[0] is not None and isinstance(jvmargs[0], str) and not jvmargs[0].startswith('-'): + if jvmargs[0] is None or (isinstance(jvmargs[0], str) and not jvmargs[0].startswith('-')): if jvmpath: raise TypeError('jvmpath specified twice') jvmpath = jvmargs[0] diff --git a/test/jpypetest/test_startup.py b/test/jpypetest/test_startup.py index 0dd609a9c..ca99424e2 100644 --- a/test/jpypetest/test_startup.py +++ b/test/jpypetest/test_startup.py @@ -16,12 +16,10 @@ # # ***************************************************************************** import jpype -import common import subrun import functools import os from pathlib import Path -import sys import unittest @@ -101,13 +99,21 @@ def testClasspathBadType(self): def testJVMPathArg_Str(self): runStartJVMTest(self.jvmpath, classpath=cp, convertStrings=False) + def testJVMPathArg_None(self): + # It is allowed to pass None as a JVM path + runStartJVMTest(None, classpath=cp, ) + + def testJVMPathArg_NoArgs(self): + runStartJVMTest(classpath=cp) + def testJVMPathArg_Path(self): with self.assertRaises(TypeError): - runStartJVMTest([ + runStartJVMTest( # Pass a path as the first argument. This isn't supported (this is # reflected in the type definition), but the fact that it "works" # gives rise to this test. - Path(self.jvmpath), cp], # type: ignore + Path(self.jvmpath), # type: ignore + classpath=cp, convertStrings=False, ) From 507ede7b53de355b33eca25abb02a9ff84ae46f0 Mon Sep 17 00:00:00 2001 From: Phil Elson Date: Wed, 3 May 2023 10:23:32 +0200 Subject: [PATCH 2/2] Systematically check that jpype.startJVM is tested and correctly typed --- jpype/_core.py | 4 +- jpype/_core.pyi | 2 - test/jpypetest/test_startup.py | 73 ++++++++++++++++++++-------------- 3 files changed, 46 insertions(+), 33 deletions(-) delete mode 100644 jpype/_core.pyi diff --git a/jpype/_core.py b/jpype/_core.py index 07f246c5d..8af9ab8f8 100644 --- a/jpype/_core.py +++ b/jpype/_core.py @@ -160,7 +160,7 @@ def interactive(): def startJVM( *jvmargs: str, jvmpath: typing.Optional[_PathOrStr] = None, - classpath: typing.Optional[_PathOrStr] = None, + classpath: typing.Optional[typing.Sequence[_PathOrStr], _PathOrStr] = None, ignoreUnrecognized: bool = False, convertStrings: bool = False, interrupt: bool = not interactive(), @@ -223,7 +223,7 @@ def startJVM( # Allow the path to be a PathLike. jvmpath = os.fspath(jvmpath) - extra_jvm_args = tuple() + extra_jvm_args: typing.Tuple[str, ...] = tuple() # Classpath handling if _hasClassPath(jvmargs): diff --git a/jpype/_core.pyi b/jpype/_core.pyi deleted file mode 100644 index f4eb32c79..000000000 --- a/jpype/_core.pyi +++ /dev/null @@ -1,2 +0,0 @@ -class _JRuntime: - pass diff --git a/test/jpypetest/test_startup.py b/test/jpypetest/test_startup.py index ca99424e2..350971f09 100644 --- a/test/jpypetest/test_startup.py +++ b/test/jpypetest/test_startup.py @@ -22,16 +22,6 @@ from pathlib import Path import unittest - -@functools.wraps(jpype.startJVM) -def runStartJVMTest(*args, **kwargs): - jpype.startJVM(*args, **kwargs) - try: - assert jpype.JClass('jpype.array.TestArray') is not None - except Exception as err: - raise RuntimeError("Test class not found") from err - - root = os.path.dirname(os.path.abspath(os.path.dirname(__file__))) cp = os.path.join(root, 'classes').replace('\\', '/') @@ -62,26 +52,39 @@ def testInvalidArgsFalse(self): def testInvalidArgsTrue(self): jpype.startJVM( "-for_sure_InVaLiD", - ignoreUnrecognized=True, convertStrings=False, + ignoreUnrecognized=True, + convertStrings=False, ) def testClasspathArgKeyword(self): - runStartJVMTest(classpath=cp, convertStrings=False) + jpype.startJVM(classpath=cp, convertStrings=False) + assert jpype.JClass('jpype.array.TestArray') is not None def testClasspathArgList(self): - runStartJVMTest(classpath=[cp], convertStrings=False) + jpype.startJVM( + classpath=[cp], + convertStrings=False, + ) + assert jpype.JClass('jpype.array.TestArray') is not None def testClasspathArgListEmpty(self): - runStartJVMTest(classpath=[cp, ''], convertStrings=False) + jpype.startJVM( + classpath=[cp, ''], + convertStrings=False, + ) + assert jpype.JClass('jpype.array.TestArray') is not None def testClasspathArgDef(self): - runStartJVMTest('-Djava.class.path=%s' % cp, convertStrings=False) + jpype.startJVM('-Djava.class.path=%s' % cp, convertStrings=False) + assert jpype.JClass('jpype.array.TestArray') is not None def testClasspathArgPath(self): - runStartJVMTest(classpath=Path(cp), convertStrings=False) + jpype.startJVM(classpath=Path(cp), convertStrings=False) + assert jpype.JClass('jpype.array.TestArray') is not None def testClasspathArgPathList(self): - runStartJVMTest(classpath=[Path(cp)], convertStrings=False) + jpype.startJVM(classpath=[Path(cp)], convertStrings=False) + assert jpype.JClass('jpype.array.TestArray') is not None def testClasspathArgGlob(self): jpype.startJVM(classpath=os.path.join(cp, '..', 'jar', 'mrjar*')) @@ -89,40 +92,52 @@ def testClasspathArgGlob(self): def testClasspathTwice(self): with self.assertRaises(TypeError): - runStartJVMTest('-Djava.class.path=%s' % + jpype.startJVM('-Djava.class.path=%s' % cp, classpath=cp, convertStrings=False) def testClasspathBadType(self): with self.assertRaises(TypeError): - runStartJVMTest(classpath=1, convertStrings=False) + jpype.startJVM(classpath=1, convertStrings=False) def testJVMPathArg_Str(self): - runStartJVMTest(self.jvmpath, classpath=cp, convertStrings=False) + jpype.startJVM(self.jvmpath, classpath=cp, convertStrings=False) + assert jpype.JClass('jpype.array.TestArray') is not None def testJVMPathArg_None(self): # It is allowed to pass None as a JVM path - runStartJVMTest(None, classpath=cp, ) + jpype.startJVM( + None, # type: ignore + classpath=cp, + ) + assert jpype.JClass('jpype.array.TestArray') is not None def testJVMPathArg_NoArgs(self): - runStartJVMTest(classpath=cp) + jpype.startJVM( + classpath=cp, + ) + assert jpype.JClass('jpype.array.TestArray') is not None def testJVMPathArg_Path(self): with self.assertRaises(TypeError): - runStartJVMTest( + jpype.startJVM( # Pass a path as the first argument. This isn't supported (this is # reflected in the type definition), but the fact that it "works" # gives rise to this test. - Path(self.jvmpath), # type: ignore - classpath=cp, + Path(self.jvmpath), # type: ignore convertStrings=False, ) def testJVMPathKeyword_str(self): - runStartJVMTest(classpath=cp, jvmpath=self.jvmpath, - convertStrings=False) + jpype.startJVM( + classpath=cp, + jvmpath=self.jvmpath, + convertStrings=False, + ) + assert jpype.JClass('jpype.array.TestArray') is not None def testJVMPathKeyword_Path(self): - runStartJVMTest(jvmpath=Path(self.jvmpath), classpath=cp, convertStrings=False) + jpype.startJVM(jvmpath=Path(self.jvmpath), classpath=cp, convertStrings=False) + assert jpype.JClass('jpype.array.TestArray') is not None def testPathTwice(self): with self.assertRaises(TypeError): @@ -130,4 +145,4 @@ def testPathTwice(self): def testBadKeyword(self): with self.assertRaises(TypeError): - jpype.startJVM(invalid=True) + jpype.startJVM(invalid=True) # type: ignore