Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

handle non ascii classpath with system classloader #1226

Merged
merged 6 commits into from
Nov 5, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
93 changes: 66 additions & 27 deletions jpype/_core.py
Original file line number Diff line number Diff line change
Expand Up @@ -32,8 +32,6 @@

from ._jvmfinder import *

from jpype._classpath import addClassPath

# This import is required to bootstrap importlib, _jpype uses importlib.util
# but on some systems it may not load properly from C. To make sure it gets
# loaded properly we are going to force the issue even through we don't
Expand Down Expand Up @@ -111,28 +109,35 @@
return _jpype.isStarted()


def _hasClassPath(args) -> bool:
def _getOldClassPath(args) -> list[str]:
for i in args:
if isinstance(i, str) and i.startswith('-Djava.class.path'):
if not isinstance(i, str):
continue
_, _, classpath = i.partition('-Djava.class.path=')
if classpath:
return classpath.split(_classpath._SEP)
return []


def _hasSystemClassLoader(args) -> bool:
for i in args:
if isinstance(i, str) and i.startswith('-Djava.system.class.loader'):
return True
return False


def _handleClassPath(
classpath: typing.Union[typing.Sequence[_PathOrStr], _PathOrStr, None] = None,
ascii: bool = True
classpath: typing.Union[typing.Sequence[_PathOrStr], _PathOrStr, None] = None
) -> typing.Sequence[str]:
"""
Return a classpath which represents the given tuple of classpath specifications
"""
out: list[str] = []
if classpath is None:
return out
if isinstance(classpath, (str, os.PathLike)):
classpath = (classpath,)
try:
# Convert anything iterable into a tuple.
classpath = tuple(classpath)
classpath = tuple(classpath) # type: ignore[arg-type]
except TypeError:
raise TypeError("Unknown class path element")

Expand All @@ -152,9 +157,11 @@
out.extend(glob.glob(pth + '.jar'))
else:
out.append(pth)
if ascii:
return [i for i in out if i.isascii()]
return [i for i in out if not i.isascii()]
return out


def _removeClassPath(args) -> tuple[str]:
return tuple(arg for arg in args if not str(arg).startswith("-Djava.class.path"))


_JVM_started = False
Expand Down Expand Up @@ -215,6 +222,9 @@
if _JVM_started:
raise OSError('JVM cannot be restarted')

has_classloader = _hasSystemClassLoader(jvmargs)


# JVM path
if jvmargs:
# jvm is the first argument the first argument is a path or None
Expand All @@ -231,24 +241,51 @@
jvmpath = os.fspath(jvmpath)

# Classpath handling
if _hasClassPath(jvmargs):
old_classpath = _getOldClassPath(jvmargs)
if old_classpath:
# Old style, specified in the arguments
if classpath is not None:
# Cannot apply both styles, conflict
raise TypeError('classpath specified twice')
classpath = old_classpath
elif classpath is None:
# Not specified at all, use the default classpath.
classpath = _classpath.getClassPath()

# Handle strings and list of strings.
extra_jvm_args = []
if classpath:
cp = _classpath._SEP.join(_handleClassPath(classpath))
extra_jvm_args += ['-Djava.class.path=%s'%cp ]
extra_jvm_args: list[str] = []

supportLib = os.path.join(os.path.dirname(os.path.dirname(__file__)), "org.jpype.jar")
if not os.path.exists(supportLib):
raise RuntimeError("Unable to find org.jpype.jar support library at " + supportLib)

late_load = not has_classloader
if classpath:
cp = _classpath._SEP.join(_handleClassPath(classpath))
if cp.isascii():
# no problems
extra_jvm_args += ['-Djava.class.path=%s'%cp ]
jvmargs = _removeClassPath(jvmargs)
late_load = False
elif has_classloader:
# https://bugs.openjdk.org/browse/JDK-8079633?jql=text%20~%20%22ParseUtil%22
raise ValueError("system classloader cannot be specified with non ascii characters in the classpath")
elif supportLib.isascii():
# ok, setup the jpype system classloader and add to the path after startup
# this guarentees all classes have the same permissions as they did in the past
extra_jvm_args += [
'-Djava.system.class.loader=org.jpype.classloader.JpypeSystemClassLoader',
'-Djava.class.path=%s'%supportLib
]
jvmargs = _removeClassPath(jvmargs)
Comment on lines +273 to +280
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Should we pass the class path as a different variable and then use it. The issue I am concerned about is that of those jars that specify a service.

https://medium.com/@yureshcs/services-and-service-provider-with-java-9-modules-ddd0170749b0

In that past when we did testing is was clear that services were only recognized if they were loaded on the initial class_path. I believe it is the case for certain things like databases. Unfortunately, it was a while since I last looked at the issue so I don't have a working test case.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Should we pass the class path as a different variable and then use it. The issue I am concerned about is that of those jars that specify a service.

https://medium.com/@yureshcs/services-and-service-provider-with-java-9-modules-ddd0170749b0

In that past when we did testing is was clear that services were only recognized if they were loaded on the initial class_path. I believe it is the case for certain things like databases. Unfortunately, it was a while since I last looked at the issue so I don't have a working test case.

They are still loaded on the initial classpath (loaded by the system class loader). I can figure out how to put together a service and add a test case for it tomorrow.

Services are loaded and instantiated lazily. So as long as the paths are added to the system classpath before returning from startJVM, before the user can do anything, then we should be ok.

else:
# We are screwed no matter what we try or do.
# Unfortunately the jdk maintainers don't seem to care either.
# This bug is almost 10 years old and spans 16 jdk versions and counting.
# https://bugs.openjdk.org/browse/JDK-8079633?jql=text%20~%20%22ParseUtil%22
raise ValueError("jpype jar must be ascii to add to the system class path")

Check warning on line 286 in jpype/_core.py

View check run for this annotation

Codecov / codecov/patch

jpype/_core.py#L286

Added line #L286 was not covered by tests


extra_jvm_args += ['-javaagent:' + supportLib]

try:
Expand Down Expand Up @@ -278,20 +315,22 @@
raise RuntimeError(f"{jvmpath} is older than required Java version{version}") from ex
raise

"""Prior versions of JPype used the jvmargs to setup the class paths via
"""Prior versions of JPype used the jvmargs to setup the class paths via
JNI (Java Native Interface) option strings:
i.e -Djava.class.path=...
i.e -Djava.class.path=...
See: https://docs.oracle.com/javase/7/docs/technotes/guides/jni/spec/invocation.html

Unfortunately, unicode is unsupported by this interface on windows, since
windows uses wide-byte (16bit) character encoding.
See: https://stackoverflow.com/questions/20052455/jni-start-jvm-with-unicode-support

To resolve this issue we add the classpath after initialization since jpype
itself supports unicode class paths.
Unfortunately, only ascii paths work because url encoding is not handled correctly
see: https://bugs.openjdk.org/browse/JDK-8079633?jql=text%20~%20%22ParseUtil%22

To resolve this issue, we add the classpath after initialization since Java has
had the utilities to correctly encode it since 1.0
"""
for cp in _handleClassPath(classpath, False):
addClassPath(Path.cwd() / Path(cp).resolve())
if late_load and classpath:
# now we can add to the system classpath
cl = _jpype.JClass("java.lang.ClassLoader").getSystemClassLoader()
for cp in _handleClassPath(classpath):
cl.addPath(_jpype._java_lang_String(cp))


def initializeResources():
Expand Down
43 changes: 43 additions & 0 deletions native/java/org/jpype/classloader/JpypeSystemClassLoader.java
Original file line number Diff line number Diff line change
@@ -0,0 +1,43 @@
/* ****************************************************************************
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at

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.

See NOTICE file for details.
**************************************************************************** */
package org.jpype.classloader;

import java.net.URL;
import java.net.URLClassLoader;
import java.nio.file.Paths;

public final class JpypeSystemClassLoader extends URLClassLoader {

public JpypeSystemClassLoader(ClassLoader parent) throws Throwable {
super(new URL[0], parent);
}

public void addPath(String path) throws Throwable {
addURL(Paths.get(path).toAbsolutePath().toUri().toURL());
}

public void addPaths(String[] paths) throws Throwable {
for (String path : paths) {
addPath(path);
}
}

// this is required to add a Java agent even if it is already in the path
@SuppressWarnings("unused")
private void appendToClassPathForInstrumentation(String path) throws Throwable {
addPath(path);
}
}
1 change: 1 addition & 0 deletions project/jars/unicode_à😎/service/README
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
Test to verify that service providers are found and used when placed in a non ascii path
20 changes: 20 additions & 0 deletions project/jars/unicode_à😎/service/build.gradle
Original file line number Diff line number Diff line change
@@ -0,0 +1,20 @@
apply plugin: 'java'
apply plugin: 'eclipse'

sourceCompatibility = 8
targetCompatibility = 8


tasks.withType(JavaCompile) {
options.release = 8
options.debug = false
}

jar {
destinationDirectory = file('dist')
from ('./src/main/java') {
include 'META-INF/services/java.time.zone.ZoneRulesProvider'
}
}

build.dependsOn(jar)
Original file line number Diff line number Diff line change
@@ -0,0 +1,28 @@
package org.jpype.service;

import java.time.ZoneId;
import java.time.zone.ZoneRules;
import java.time.zone.ZoneRulesProvider;
import java.util.Collections;
import java.util.NavigableMap;
import java.util.Set;
import java.util.TreeMap;

public class JpypeZoneRulesProvider extends ZoneRulesProvider {

@Override
protected Set<String> provideZoneIds() {
return Collections.singleton("JpypeTest/Timezone");
}

@Override
protected ZoneRules provideRules(String zoneId, boolean forCaching) {
return ZoneId.of("UTC").getRules();
}

@Override
protected NavigableMap<String, ZoneRules> provideVersions(String zoneId) {
return new TreeMap<>();
}

}
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
org.jpype.service.JpypeZoneRulesProvider
42 changes: 42 additions & 0 deletions test/harness/jpype/startup/TestSystemClassLoader.java
Original file line number Diff line number Diff line change
@@ -0,0 +1,42 @@
/* ****************************************************************************
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at

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.

See NOTICE file for details.
**************************************************************************** */
package jpype.startup;

import java.net.URL;
import java.net.URLClassLoader;
import java.nio.file.Paths;

public class TestSystemClassLoader extends URLClassLoader {

public TestSystemClassLoader(ClassLoader parent) throws Throwable {
super(new URL[0], parent);
}

public void addPath(String path) throws Throwable {
addURL(Paths.get(path).toAbsolutePath().toUri().toURL());
}

public void addPaths(String[] paths) throws Throwable {
for (String path : paths) {
addPath(path);
}
}

@SuppressWarnings("unused") // needed to start with agent
private void appendToClassPathForInstrumentation(String path) throws Throwable {
addPath(path);
}
}
Binary file added test/jar/unicode_à😎/service.jar
Binary file not shown.
69 changes: 69 additions & 0 deletions test/jpypetest/test_startup.py
Original file line number Diff line number Diff line change
Expand Up @@ -151,4 +151,73 @@ def testNonASCIIPath(self):
Regression test for https://github.com/jpype-project/jpype/issues/1194
"""
jpype.startJVM(jvmpath=Path(self.jvmpath), classpath="test/jar/unicode_à😎/sample_package.jar")
cl = jpype.JClass("java.lang.ClassLoader").getSystemClassLoader()
self.assertEqual(type(cl), jpype.JClass("org.jpype.classloader.JpypeSystemClassLoader"))
assert dir(jpype.JPackage('org.jpype.sample_package')) == ['A', 'B']


def testOldStyleNonASCIIPath(self):
"""Test that paths with non-ASCII characters are handled correctly.
Regression test for https://github.com/jpype-project/jpype/issues/1194
"""
jpype.startJVM("-Djava.class.path=test/jar/unicode_à😎/sample_package.jar", jvmpath=Path(self.jvmpath))
cl = jpype.JClass("java.lang.ClassLoader").getSystemClassLoader()
self.assertEqual(type(cl), jpype.JClass("org.jpype.classloader.JpypeSystemClassLoader"))
assert dir(jpype.JPackage('org.jpype.sample_package')) == ['A', 'B']

def testNonASCIIPathWithSystemClassLoader(self):
with self.assertRaises(ValueError):
jpype.startJVM(
"-Djava.system.class.loader=jpype.startup.TestSystemClassLoader",
jvmpath=Path(self.jvmpath),
classpath="test/jar/unicode_à😎/sample_package.jar"
)

def testOldStyleNonASCIIPathWithSystemClassLoader(self):
with self.assertRaises(ValueError):
jpype.startJVM(
self.jvmpath,
"-Djava.system.class.loader=jpype.startup.TestSystemClassLoader",
"-Djava.class.path=test/jar/unicode_à😎/sample_package.jar"
)

def testASCIIPathWithSystemClassLoader(self):
jpype.startJVM(
"-Djava.system.class.loader=jpype.startup.TestSystemClassLoader",
jvmpath=Path(self.jvmpath),
classpath=f"test/classes"
)
classloader = jpype.JClass("java.lang.ClassLoader").getSystemClassLoader()
test_classLoader = jpype.JClass("jpype.startup.TestSystemClassLoader")
self.assertEqual(type(classloader), test_classLoader)
assert dir(jpype.JPackage('jpype.startup')) == ['TestSystemClassLoader']

def testOldStyleASCIIPathWithSystemClassLoader(self):
jpype.startJVM(
self.jvmpath,
"-Djava.system.class.loader=jpype.startup.TestSystemClassLoader",
"-Djava.class.path=test/classes"
)
classloader = jpype.JClass("java.lang.ClassLoader").getSystemClassLoader()
test_classLoader = jpype.JClass("jpype.startup.TestSystemClassLoader")
self.assertEqual(type(classloader), test_classLoader)
assert dir(jpype.JPackage('jpype.startup')) == ['TestSystemClassLoader']

def testDefaultSystemClassLoader(self):
# we introduce no behavior change unless absolutely necessary
jpype.startJVM(jvmpath=Path(self.jvmpath))
cl = jpype.JClass("java.lang.ClassLoader").getSystemClassLoader()
self.assertNotEqual(type(cl), jpype.JClass("org.jpype.classloader.JpypeSystemClassLoader"))

def testServiceWithNonASCIIPath(self):
jpype.startJVM(
self.jvmpath,
"-Djava.locale.providers=SPI,CLDR",
classpath="test/jar/unicode_à😎/service.jar",
)
ZoneId = jpype.JClass("java.time.ZoneId")
ZoneRulesException = jpype.JClass("java.time.zone.ZoneRulesException")
try:
ZoneId.of("JpypeTest/Timezone")
except ZoneRulesException:
self.fail("JpypeZoneRulesProvider not loaded")
4 changes: 2 additions & 2 deletions test/jpypetest/test_thread.py
Original file line number Diff line number Diff line change
Expand Up @@ -16,11 +16,11 @@
#
# *****************************************************************************
import jpype
import sys
import time
import common
import pytest

from jpype.imports import *


class ThreadTestCase(common.JPypeTestCase):
def setUp(self):
Expand Down
Loading