mirror of https://github.com/kivy/pyjnius.git
1060 lines
37 KiB
Cython
1060 lines
37 KiB
Cython
from cpython cimport PyObject
|
|
from warnings import warn
|
|
|
|
class JavaException(Exception):
|
|
'''Can be a real java exception, or just an exception from the wrapper.
|
|
'''
|
|
classname = None # The classname of the exception
|
|
innermessage = None # The message of the inner exception
|
|
stacktrace = None # The stack trace of the inner exception
|
|
|
|
def __init__(self, message, classname=None, innermessage=None, stacktrace=None):
|
|
self.classname = classname
|
|
self.innermessage = innermessage
|
|
self.stacktrace = stacktrace
|
|
Exception.__init__(self, message)
|
|
|
|
|
|
cdef class JavaObject(object):
|
|
'''Can contain any Java object. Used to store instance, or whatever.
|
|
'''
|
|
|
|
cdef jobject obj
|
|
|
|
def __cinit__(self):
|
|
self.obj = NULL
|
|
|
|
|
|
cdef class JavaClassStorage:
|
|
cdef jclass j_cls
|
|
|
|
def __cinit__(self):
|
|
self.j_cls = NULL
|
|
|
|
def __dealloc__(self):
|
|
cdef JNIEnv *j_env
|
|
if self.j_cls != NULL:
|
|
j_env = get_jnienv()
|
|
j_env[0].DeleteGlobalRef(j_env, self.j_cls)
|
|
self.j_cls = NULL
|
|
|
|
|
|
class MetaJavaBase(type):
|
|
def __instancecheck__(cls, value):
|
|
cdef JNIEnv *j_env = get_jnienv()
|
|
cdef JavaClassStorage meta = getattr(cls, '__cls_storage', None)
|
|
cdef JavaObject jo
|
|
cdef JavaClass jc
|
|
cdef PythonJavaClass pc
|
|
cdef jobject obj = NULL
|
|
cdef jclass proxy = j_env[0].FindClass(j_env, <char *>'java/lang/reflect/Proxy')
|
|
cdef jclass nih
|
|
cdef jmethodID meth
|
|
cdef object wrapped_python
|
|
|
|
if isinstance(value, base_string):
|
|
obj = j_env[0].NewStringUTF(j_env, <char *>"")
|
|
elif isinstance(value, JavaClass):
|
|
jc = value
|
|
obj = jc.j_self.obj
|
|
elif isinstance(value, JavaObject):
|
|
jo = value
|
|
obj = jo.obj
|
|
elif isinstance(value, PythonJavaClass):
|
|
pc = value
|
|
jc = pc.j_self
|
|
if jc is None:
|
|
pc._init_j_self_ptr()
|
|
jc = pc.j_self
|
|
obj = jc.j_self.obj
|
|
|
|
if NULL != obj:
|
|
if meta is not None and 0 != j_env[0].IsInstanceOf(j_env, obj, meta.j_cls):
|
|
return True
|
|
|
|
if NULL != proxy and 0 != j_env[0].IsInstanceOf(j_env, obj, proxy):
|
|
# value is a proxy object. check whether it's one of ours
|
|
meth = j_env[0].GetStaticMethodID(
|
|
j_env, proxy, <char *>'getInvocationHandler',
|
|
<char *>'(Ljava/lang/Object;)Ljava/lang/reflect/InvocationHandler;'
|
|
)
|
|
obj = j_env[0].CallStaticObjectMethod(j_env, proxy, meth, obj)
|
|
nih = j_env[0].FindClass(j_env, <char *>'org/jnius/NativeInvocationHandler')
|
|
if NULL == nih:
|
|
# nih is not reliably in the classpath. don't crash if it's
|
|
# not there, because it's impossible to get this far with
|
|
# a PythonJavaClass without it, so we can safely assume this
|
|
# is just a POJO from elsewhere.
|
|
j_env[0].ExceptionClear(j_env)
|
|
else:
|
|
meth = j_env[0].GetMethodID(
|
|
j_env, nih, <char *>'getPythonObjectPointer',
|
|
<char *>'()J'
|
|
)
|
|
if NULL == meth:
|
|
# Perhaps we have an old nih
|
|
j_env[0].ExceptionClear(j_env)
|
|
warn("The org.jnius.NativeInvocationHandler on your classpath"
|
|
" is out of date. isinstance will be unreliable.")
|
|
else:
|
|
wrapped_python = <object><PyObject *>j_env[0].CallLongMethod(j_env, obj, meth)
|
|
if wrapped_python is not value and wrapped_python is not None:
|
|
if isinstance(wrapped_python, cls):
|
|
return True
|
|
|
|
# All else fails, defer to python.
|
|
return super(MetaJavaBase, cls).__instancecheck__(value)
|
|
|
|
|
|
cdef dict jclass_register = {}
|
|
|
|
|
|
class MetaJavaClass(MetaJavaBase):
|
|
def __new__(meta, classname, bases, classDict):
|
|
meta.resolve_class(classDict)
|
|
tp = type.__new__(meta, str(classname), bases, classDict)
|
|
jclass_register[classDict['__javaclass__']] = tp
|
|
return tp
|
|
|
|
def __subclasscheck__(cls, value):
|
|
cdef JNIEnv *j_env = get_jnienv()
|
|
cdef JavaClassStorage me = getattr(cls, '__cls_storage')
|
|
cdef JavaClassStorage jcs
|
|
cdef JavaClass jc
|
|
cdef jclass obj = NULL
|
|
|
|
if isinstance(value, JavaClass):
|
|
jc = value
|
|
obj = jc.j_self.obj
|
|
else:
|
|
jcs = getattr(value, '__cls_storage', None)
|
|
if jcs is not None:
|
|
obj = jcs.j_cls
|
|
|
|
if NULL == obj:
|
|
for interface in getattr(value, '__javainterfaces__', []):
|
|
obj = j_env[0].FindClass(j_env, str_for_c(interface))
|
|
if obj == NULL:
|
|
j_env[0].ExceptionClear(j_env)
|
|
elif 0 != j_env[0].IsAssignableFrom(j_env, obj, me.j_cls):
|
|
return True
|
|
else:
|
|
if 0 != j_env[0].IsAssignableFrom(j_env, obj, me.j_cls):
|
|
return True
|
|
|
|
return super(MetaJavaClass, cls).__subclasscheck__(value)
|
|
|
|
@staticmethod
|
|
def get_javaclass(name):
|
|
return jclass_register.get(name)
|
|
|
|
@classmethod
|
|
def resolve_class(meta, classDict):
|
|
# search the Java class, and bind to our object
|
|
if not '__javaclass__' in classDict:
|
|
raise JavaException('__javaclass__ definition missing')
|
|
|
|
cdef JavaClassStorage jcs = JavaClassStorage()
|
|
cdef bytes __javaclass__ = <bytes>classDict['__javaclass__']
|
|
cdef bytes __javainterfaces__ = <bytes>classDict.get('__javainterfaces__', b'')
|
|
cdef bytes __javabaseclass__ = <bytes>classDict.get('__javabaseclass__', b'')
|
|
cdef jmethodID getProxyClass, getClassLoader
|
|
cdef jclass *interfaces
|
|
cdef jobject *jargs
|
|
cdef JNIEnv *j_env = get_jnienv()
|
|
|
|
if __javainterfaces__ and __javabaseclass__:
|
|
baseclass = j_env[0].FindClass(j_env, <char*>__javabaseclass__)
|
|
interfaces = <jclass *>malloc(sizeof(jclass) * len(__javainterfaces__))
|
|
|
|
for n, i in enumerate(__javainterfaces__):
|
|
interfaces[n] = j_env[0].FindClass(j_env, <char*>i)
|
|
|
|
getProxyClass = j_env[0].GetStaticMethodID(
|
|
j_env, baseclass, "getProxyClass",
|
|
"(Ljava/lang/ClassLoader,[Ljava/lang/Class;)Ljava/lang/Class;")
|
|
|
|
getClassLoader = j_env[0].GetStaticMethodID(
|
|
j_env, baseclass, "getClassLoader", "()Ljava/lang/Class;")
|
|
|
|
with nogil:
|
|
classLoader = j_env[0].CallStaticObjectMethodA(
|
|
j_env, baseclass, getClassLoader, NULL)
|
|
jargs = <jobject *>malloc(sizeof(jobject) * 2)
|
|
jargs[0] = <jobject *>classLoader
|
|
jargs[1] = interfaces
|
|
jcs.j_cls = j_env[0].CallStaticObjectMethod(
|
|
j_env, baseclass, getProxyClass, jargs)
|
|
|
|
j_env[0].DeleteLocalRef(j_env, baseclass)
|
|
|
|
if jcs.j_cls == NULL:
|
|
raise JavaException('Unable to create the class'
|
|
' {0}'.format(__javaclass__))
|
|
else:
|
|
class_name = str_for_c(__javaclass__)
|
|
jcs.j_cls = j_env[0].FindClass(j_env,
|
|
<char *>class_name)
|
|
if jcs.j_cls == NULL:
|
|
raise JavaException('Unable to find the class'
|
|
' {0}'.format(__javaclass__))
|
|
|
|
# XXX do we need to grab a ref here?
|
|
# -> Yes, according to http://developer.android.com/training/articles/perf-jni.html
|
|
# in the section Local and Global References
|
|
jcs.j_cls = j_env[0].NewGlobalRef(j_env, jcs.j_cls)
|
|
|
|
classDict['__cls_storage'] = jcs
|
|
|
|
# search all the static JavaMethod within our class, and resolve them
|
|
cdef JavaMethod jm
|
|
cdef JavaMultipleMethod jmm
|
|
for name, value in items_compat(classDict):
|
|
if isinstance(value, JavaMethod):
|
|
jm = value
|
|
if not jm.is_static:
|
|
continue
|
|
jm.set_resolve_info(j_env, jcs.j_cls, None,
|
|
str_for_c(name), str_for_c(__javaclass__))
|
|
elif isinstance(value, JavaMultipleMethod):
|
|
jmm = value
|
|
jmm.set_resolve_info(j_env, jcs.j_cls, None,
|
|
str_for_c(name), str_for_c(__javaclass__))
|
|
|
|
|
|
# search all the static JavaField within our class, and resolve them
|
|
cdef JavaField jf
|
|
for name, value in items_compat(classDict):
|
|
if not isinstance(value, JavaField):
|
|
continue
|
|
jf = value
|
|
if not jf.is_static:
|
|
continue
|
|
jf.set_resolve_info(j_env, jcs.j_cls,
|
|
str_for_c(name), str_for_c(__javaclass__))
|
|
|
|
|
|
cdef class JavaClass(object):
|
|
'''Main class to do introspection.
|
|
'''
|
|
|
|
cdef JNIEnv *j_env
|
|
cdef jclass j_cls
|
|
cdef LocalRef j_self
|
|
|
|
def __cinit__(self, *args, **kwargs):
|
|
self.j_cls = NULL
|
|
self.j_self = None
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
super(JavaClass, self).__init__()
|
|
# copy the current attribute in the storage to our class
|
|
cdef JavaClassStorage jcs = self.__cls_storage
|
|
self.j_cls = jcs.j_cls
|
|
|
|
if 'noinstance' not in kwargs:
|
|
self.call_constructor(args)
|
|
self.resolve_methods()
|
|
self.resolve_fields()
|
|
|
|
cdef void instanciate_from(self, LocalRef j_self) except *:
|
|
self.j_self = j_self
|
|
self.resolve_methods()
|
|
self.resolve_fields()
|
|
|
|
cdef void call_constructor(self, args) except *:
|
|
# the goal is to find the class constructor, and call it with the
|
|
# correct arguments.
|
|
cdef jvalue *j_args = NULL
|
|
cdef jobject j_self = NULL
|
|
cdef jmethodID constructor = NULL
|
|
cdef JNIEnv *j_env = get_jnienv()
|
|
cdef list found_definitions = []
|
|
|
|
# get the constructor definition if exist
|
|
definitions = [('()V', False)]
|
|
if hasattr(self, '__javaconstructor__'):
|
|
definitions = self.__javaconstructor__
|
|
if isinstance(definitions, base_string):
|
|
definitions = [definitions]
|
|
|
|
if len(definitions) == 0:
|
|
raise JavaException('No constructor available')
|
|
|
|
elif len(definitions) == 1:
|
|
definition, is_varargs = definitions[0]
|
|
found_definitions = [definition]
|
|
d_ret, d_args = parse_definition(definition)
|
|
|
|
if is_varargs:
|
|
args_ = args[:len(d_args) - 1] + (args[len(d_args) - 1:],)
|
|
else:
|
|
args_ = args
|
|
if len(args or ()) != len(d_args or ()):
|
|
raise JavaException(
|
|
'Invalid call, number of argument mismatch for '
|
|
'constructor, available: {}'.format(found_definitions)
|
|
)
|
|
else:
|
|
scores = []
|
|
for definition, is_varargs in definitions:
|
|
found_definitions.append(definition)
|
|
d_ret, d_args = parse_definition(definition)
|
|
if is_varargs:
|
|
args_ = args[:len(d_args) - 1] + (args[len(d_args) - 1:],)
|
|
else:
|
|
args_ = args
|
|
|
|
score = calculate_score(d_args, args)
|
|
if score == -1:
|
|
continue
|
|
scores.append((score, definition, d_ret, d_args, args_))
|
|
if not scores:
|
|
raise JavaException(
|
|
'No constructor matching your arguments, available: '
|
|
'{}'.format(found_definitions)
|
|
)
|
|
scores.sort()
|
|
score, definition, d_ret, d_args, args_ = scores[-1]
|
|
|
|
try:
|
|
# convert python arguments to java arguments
|
|
if len(args):
|
|
j_args = <jvalue *>malloc(sizeof(jvalue) * len(d_args))
|
|
if j_args == NULL:
|
|
raise MemoryError('Unable to allocate memory for java args')
|
|
populate_args(j_env, d_args, j_args, args_)
|
|
|
|
# get the java constructor
|
|
defstr = str_for_c(definition)
|
|
constructor = j_env[0].GetMethodID(
|
|
j_env, self.j_cls, '<init>', <char *><bytes>defstr)
|
|
if constructor == NULL:
|
|
raise JavaException('Unable to found the constructor'
|
|
' for {0}'.format(self.__javaclass__))
|
|
|
|
# create the object
|
|
j_self = j_env[0].NewObjectA(j_env, self.j_cls,
|
|
constructor, j_args)
|
|
|
|
# release our arguments
|
|
release_args(j_env, d_args, j_args, args_)
|
|
|
|
check_exception(j_env)
|
|
if j_self == NULL:
|
|
raise JavaException('Unable to instanciate {0}'.format(
|
|
self.__javaclass__))
|
|
|
|
self.j_self = create_local_ref(j_env, j_self)
|
|
j_env[0].DeleteLocalRef(j_env, j_self)
|
|
finally:
|
|
# in case NewObjectA() throws an exception,
|
|
# the execution might not get further, but 'finally' block
|
|
# will still be called
|
|
check_exception(j_env)
|
|
if j_args != NULL:
|
|
free(j_args)
|
|
|
|
cdef void resolve_methods(self) except *:
|
|
# search all the JavaMethod within our class, and resolve them
|
|
cdef JavaMethod jm
|
|
cdef JavaMultipleMethod jmm
|
|
cdef JNIEnv *j_env = get_jnienv()
|
|
for name, value in items_compat(self.__class__.__dict__):
|
|
if isinstance(value, JavaMethod):
|
|
jm = value
|
|
if jm.is_static:
|
|
continue
|
|
jm.set_resolve_info(j_env, self.j_cls, self.j_self,
|
|
str_for_c(name), str_for_c(self.__javaclass__))
|
|
elif isinstance(value, JavaMultipleMethod):
|
|
jmm = value
|
|
jmm.set_resolve_info(j_env, self.j_cls, self.j_self,
|
|
str_for_c(name), str_for_c(self.__javaclass__))
|
|
|
|
cdef void resolve_fields(self) except *:
|
|
# search all the JavaField within our class, and resolve them
|
|
cdef JavaField jf
|
|
cdef JNIEnv *j_env = get_jnienv()
|
|
for name, value in items_compat(self.__class__.__dict__):
|
|
if not isinstance(value, JavaField):
|
|
continue
|
|
jf = value
|
|
if jf.is_static:
|
|
continue
|
|
jf.set_resolve_info(j_env, self.j_cls,
|
|
name, self.__javaclass__)
|
|
|
|
def __repr__(self):
|
|
return '<{0} at 0x{1:x} jclass={2} jself={3}>'.format(
|
|
self.__class__.__name__,
|
|
id(self),
|
|
self.__javaclass__,
|
|
self.j_self)
|
|
|
|
|
|
cdef class JavaField(object):
|
|
cdef jfieldID j_field
|
|
cdef JNIEnv *j_env
|
|
cdef jclass j_cls
|
|
cdef object is_static
|
|
cdef name
|
|
cdef classname
|
|
cdef definition
|
|
|
|
def __cinit__(self, definition, **kwargs):
|
|
self.j_field = NULL
|
|
self.j_cls = NULL
|
|
|
|
def __init__(self, definition, **kwargs):
|
|
super(JavaField, self).__init__()
|
|
self.definition = definition
|
|
self.is_static = kwargs.get('static', False)
|
|
|
|
cdef void set_resolve_info(self, JNIEnv *j_env, jclass j_cls,
|
|
name, classname):
|
|
j_env = get_jnienv()
|
|
self.name = name
|
|
self.classname = classname
|
|
self.j_cls = j_cls
|
|
|
|
cdef void ensure_field(self) except *:
|
|
cdef JNIEnv *j_env = get_jnienv()
|
|
if self.j_field != NULL:
|
|
return
|
|
if self.is_static:
|
|
defstr = str_for_c(self.definition)
|
|
self.j_field = j_env[0].GetStaticFieldID(
|
|
j_env, self.j_cls, <char *>self.name,
|
|
<char *>defstr
|
|
)
|
|
else:
|
|
defstr = str_for_c(self.definition)
|
|
namestr = str_for_c(self.name)
|
|
self.j_field = j_env[0].GetFieldID(
|
|
j_env, self.j_cls, <char *>namestr,
|
|
<char *>defstr
|
|
)
|
|
if self.j_field == NULL:
|
|
raise JavaException(
|
|
'Unable to find the field {0}'.format(self.name)
|
|
)
|
|
|
|
def __get__(self, obj, objtype):
|
|
cdef jobject j_self
|
|
|
|
self.ensure_field()
|
|
if obj is None:
|
|
return self.read_static_field()
|
|
|
|
j_self = (<JavaClass?>obj).j_self.obj
|
|
return self.read_field(j_self)
|
|
|
|
def __set__(self, obj, value):
|
|
cdef jobject j_self
|
|
|
|
self.ensure_field()
|
|
if obj is None:
|
|
# set not implemented for static fields
|
|
raise NotImplementedError()
|
|
|
|
j_self = (<JavaClass?>obj).j_self.obj
|
|
self.write_field(j_self, value)
|
|
|
|
cdef write_field(self, jobject j_self, value):
|
|
cdef jboolean j_boolean
|
|
cdef jbyte j_byte
|
|
cdef jchar j_char
|
|
cdef jshort j_short
|
|
cdef jint j_int
|
|
cdef jlong j_long
|
|
cdef jfloat j_float
|
|
cdef jdouble j_double
|
|
cdef jobject j_object
|
|
cdef JNIEnv *j_env = get_jnienv()
|
|
|
|
# type of the java field
|
|
r = self.definition[0]
|
|
|
|
# set the java field; implemented only for primitive types
|
|
if r == 'Z':
|
|
j_boolean = <jboolean>value
|
|
j_env[0].SetBooleanField(j_env, j_self, self.j_field, j_boolean)
|
|
elif r == 'B':
|
|
j_byte = <jbyte>value
|
|
j_env[0].SetByteField(j_env, j_self, self.j_field, j_byte)
|
|
elif r == 'C':
|
|
j_char = <jchar>value
|
|
j_env[0].SetCharField(j_env, j_self, self.j_field, j_char)
|
|
elif r == 'S':
|
|
j_short = <jshort>value
|
|
j_env[0].SetShortField(j_env, j_self, self.j_field, j_short)
|
|
elif r == 'I':
|
|
j_int = <jint>value
|
|
j_env[0].SetIntField(j_env, j_self, self.j_field, j_int)
|
|
elif r == 'J':
|
|
j_long = <jlong>value
|
|
j_env[0].SetLongField(j_env, j_self, self.j_field, j_long)
|
|
elif r == 'F':
|
|
j_float = <jfloat>value
|
|
j_env[0].SetFloatField(j_env, j_self, self.j_field, j_float)
|
|
elif r == 'D':
|
|
j_double = <jdouble>value
|
|
j_env[0].SetDoubleField(j_env, j_self, self.j_field, j_double)
|
|
elif r == 'L':
|
|
j_object = <jobject>convert_python_to_jobject(j_env, self.definition, value)
|
|
j_env[0].SetObjectField(j_env, j_self, self.j_field, j_object)
|
|
j_env[0].DeleteLocalRef(j_env, j_object)
|
|
else:
|
|
raise Exception(
|
|
"Invalid field definition '{}'".format(r)
|
|
)
|
|
|
|
check_exception(j_env)
|
|
|
|
cdef read_field(self, jobject j_self):
|
|
cdef jboolean j_boolean
|
|
cdef jbyte j_byte
|
|
cdef jchar j_char
|
|
cdef jshort j_short
|
|
cdef jint j_int
|
|
cdef jlong j_long
|
|
cdef jfloat j_float
|
|
cdef jdouble j_double
|
|
cdef jobject j_object
|
|
cdef char *c_str
|
|
cdef bytes py_str
|
|
cdef object ret = None
|
|
cdef JavaObject ret_jobject
|
|
cdef JavaClass ret_jc
|
|
cdef JNIEnv *j_env = get_jnienv()
|
|
|
|
# return type of the java method
|
|
r = self.definition[0]
|
|
|
|
# now call the java method
|
|
if r == 'Z':
|
|
j_boolean = j_env[0].GetBooleanField(
|
|
j_env, j_self, self.j_field)
|
|
ret = True if j_boolean else False
|
|
elif r == 'B':
|
|
j_byte = j_env[0].GetByteField(
|
|
j_env, j_self, self.j_field)
|
|
ret = <char>j_byte
|
|
elif r == 'C':
|
|
j_char = j_env[0].GetCharField(
|
|
j_env, j_self, self.j_field)
|
|
ret = chr(<char>j_char)
|
|
elif r == 'S':
|
|
j_short = j_env[0].GetShortField(
|
|
j_env, j_self, self.j_field)
|
|
ret = <short>j_short
|
|
elif r == 'I':
|
|
j_int = j_env[0].GetIntField(
|
|
j_env, j_self, self.j_field)
|
|
ret = <int>j_int
|
|
elif r == 'J':
|
|
j_long = j_env[0].GetLongField(
|
|
j_env, j_self, self.j_field)
|
|
ret = <long long>j_long
|
|
elif r == 'F':
|
|
j_float = j_env[0].GetFloatField(
|
|
j_env, j_self, self.j_field)
|
|
ret = <float>j_float
|
|
elif r == 'D':
|
|
j_double = j_env[0].GetDoubleField(
|
|
j_env, j_self, self.j_field)
|
|
ret = <double>j_double
|
|
elif r == 'L':
|
|
j_object = j_env[0].GetObjectField(
|
|
j_env, j_self, self.j_field)
|
|
check_exception(j_env)
|
|
if j_object != NULL:
|
|
ret = convert_jobject_to_python(
|
|
j_env, self.definition, j_object)
|
|
j_env[0].DeleteLocalRef(j_env, j_object)
|
|
elif r == '[':
|
|
r = self.definition[1:]
|
|
j_object = j_env[0].GetObjectField(
|
|
j_env, j_self, self.j_field)
|
|
check_exception(j_env)
|
|
if j_object != NULL:
|
|
ret = convert_jarray_to_python(j_env, r, j_object)
|
|
j_env[0].DeleteLocalRef(j_env, j_object)
|
|
else:
|
|
raise Exception(
|
|
"Invalid field definition '{}'".format(r)
|
|
)
|
|
|
|
check_exception(j_env)
|
|
return ret
|
|
|
|
cdef read_static_field(self):
|
|
cdef jboolean j_boolean
|
|
cdef jbyte j_byte
|
|
cdef jchar j_char
|
|
cdef jshort j_short
|
|
cdef jint j_int
|
|
cdef jlong j_long
|
|
cdef jfloat j_float
|
|
cdef jdouble j_double
|
|
cdef jobject j_object
|
|
cdef object ret = None
|
|
cdef JNIEnv *j_env = get_jnienv()
|
|
|
|
# return type of the java method
|
|
r = self.definition[0]
|
|
|
|
# now call the java method
|
|
if r == 'Z':
|
|
j_boolean = j_env[0].GetStaticBooleanField(
|
|
j_env, self.j_cls, self.j_field)
|
|
ret = True if j_boolean else False
|
|
elif r == 'B':
|
|
j_byte = j_env[0].GetStaticByteField(
|
|
j_env, self.j_cls, self.j_field)
|
|
ret = <char>j_byte
|
|
elif r == 'C':
|
|
j_char = j_env[0].GetStaticCharField(
|
|
j_env, self.j_cls, self.j_field)
|
|
ret = chr(<char>j_char)
|
|
elif r == 'S':
|
|
j_short = j_env[0].GetStaticShortField(
|
|
j_env, self.j_cls, self.j_field)
|
|
ret = <short>j_short
|
|
elif r == 'I':
|
|
j_int = j_env[0].GetStaticIntField(
|
|
j_env, self.j_cls, self.j_field)
|
|
ret = <int>j_int
|
|
elif r == 'J':
|
|
j_long = j_env[0].GetStaticLongField(
|
|
j_env, self.j_cls, self.j_field)
|
|
ret = <long long>j_long
|
|
elif r == 'F':
|
|
j_float = j_env[0].GetStaticFloatField(
|
|
j_env, self.j_cls, self.j_field)
|
|
ret = <float>j_float
|
|
elif r == 'D':
|
|
j_double = j_env[0].GetStaticDoubleField(
|
|
j_env, self.j_cls, self.j_field)
|
|
ret = <double>j_double
|
|
elif r == 'L':
|
|
j_object = j_env[0].GetStaticObjectField(
|
|
j_env, self.j_cls, self.j_field)
|
|
check_exception(j_env)
|
|
if j_object != NULL:
|
|
ret = convert_jobject_to_python(
|
|
j_env, self.definition, j_object)
|
|
j_env[0].DeleteLocalRef(j_env, j_object)
|
|
elif r == '[':
|
|
r = self.definition[1:]
|
|
j_object = j_env[0].GetStaticObjectField(
|
|
j_env, self.j_cls, self.j_field)
|
|
check_exception(j_env)
|
|
if j_object != NULL:
|
|
ret = convert_jarray_to_python(j_env, r, j_object)
|
|
j_env[0].DeleteLocalRef(j_env, j_object)
|
|
else:
|
|
raise Exception(
|
|
"Invalid field definition '{}'".format(r)
|
|
)
|
|
|
|
check_exception(j_env)
|
|
return ret
|
|
|
|
|
|
cdef class JavaMethod(object):
|
|
'''Used to resolve a Java method, and do the call
|
|
'''
|
|
cdef jmethodID j_method
|
|
cdef jclass j_cls
|
|
cdef LocalRef j_self
|
|
cdef name
|
|
cdef classname
|
|
cdef definition
|
|
cdef object is_static
|
|
cdef bint is_varargs
|
|
cdef object definition_return
|
|
cdef object definition_args
|
|
|
|
def __cinit__(self, definition, **kwargs):
|
|
self.j_method = NULL
|
|
self.j_cls = NULL
|
|
self.j_self = None
|
|
|
|
def __init__(self, definition, **kwargs):
|
|
super(JavaMethod, self).__init__()
|
|
self.definition = definition
|
|
self.definition_return, self.definition_args = parse_definition(
|
|
definition
|
|
)
|
|
self.is_static = kwargs.get('static', False)
|
|
self.is_varargs = kwargs.get('varargs', False)
|
|
|
|
cdef void ensure_method(self) except *:
|
|
if self.j_method != NULL:
|
|
return
|
|
cdef JNIEnv *j_env = get_jnienv()
|
|
if self.name is None:
|
|
raise JavaException(
|
|
'Unable to find a None method!\nclassname: {}, definition: {}'
|
|
.format(self.classname, self.definition)
|
|
)
|
|
if self.is_static:
|
|
defstr = str_for_c(self.definition)
|
|
self.j_method = j_env[0].GetStaticMethodID(
|
|
j_env, self.j_cls, <char *>self.name,
|
|
<char *>defstr)
|
|
else:
|
|
defstr = str_for_c(self.definition)
|
|
self.j_method = j_env[0].GetMethodID(
|
|
j_env, self.j_cls, <char *>self.name,
|
|
<char *>defstr)
|
|
|
|
if self.j_method == NULL:
|
|
raise JavaException('Unable to find the method'
|
|
' {0}({1})'.format(self.name, self.definition))
|
|
|
|
cdef void set_resolve_info(self, JNIEnv *j_env, jclass j_cls,
|
|
LocalRef j_self, name, classname):
|
|
self.name = name
|
|
self.classname = classname
|
|
self.j_cls = j_cls
|
|
self.j_self = j_self
|
|
|
|
def __get__(self, obj, objtype):
|
|
if obj is None:
|
|
return self
|
|
# XXX FIXME we MUST not change our own j_self, but return a "bound"
|
|
# method here, as python does!
|
|
cdef JavaClass jc = obj
|
|
self.j_self = jc.j_self
|
|
return self
|
|
|
|
def __call__(self, *args):
|
|
# argument array to pass to the method
|
|
cdef jvalue *j_args = NULL
|
|
cdef tuple d_args = self.definition_args
|
|
cdef int d_args_len = len(d_args)
|
|
cdef JNIEnv *j_env = get_jnienv()
|
|
|
|
if self.is_varargs:
|
|
args = args[:d_args_len - 1] + (args[d_args_len - 1:],)
|
|
|
|
if len(args) != d_args_len:
|
|
raise JavaException(
|
|
'Invalid call, number of argument mismatch, '
|
|
'got {} need {}'.format(len(args), d_args_len)
|
|
)
|
|
|
|
if not self.is_static and j_env == NULL:
|
|
raise JavaException(
|
|
'Cannot call instance method on a un-instanciated class'
|
|
)
|
|
|
|
self.ensure_method()
|
|
|
|
try:
|
|
# convert python argument if necessary
|
|
if len(args):
|
|
j_args = <jvalue *>malloc(sizeof(jvalue) * d_args_len)
|
|
if j_args == NULL:
|
|
raise MemoryError('Unable to allocate memory for java args')
|
|
populate_args(j_env, self.definition_args, j_args, args)
|
|
|
|
try:
|
|
# do the call
|
|
if self.is_static:
|
|
return self.call_staticmethod(j_env, j_args)
|
|
return self.call_method(j_env, j_args)
|
|
finally:
|
|
release_args(j_env, self.definition_args, j_args, args)
|
|
|
|
finally:
|
|
if j_args != NULL:
|
|
free(j_args)
|
|
|
|
cdef call_method(self, JNIEnv *j_env, jvalue *j_args):
|
|
cdef jboolean j_boolean
|
|
cdef jbyte j_byte
|
|
cdef jchar j_char
|
|
cdef jshort j_short
|
|
cdef jint j_int
|
|
cdef jlong j_long
|
|
cdef jfloat j_float
|
|
cdef jdouble j_double
|
|
cdef jobject j_object
|
|
cdef char *c_str
|
|
cdef bytes py_str
|
|
cdef object ret = None
|
|
cdef JavaObject ret_jobject
|
|
cdef JavaClass ret_jc
|
|
cdef jobject j_self = self.j_self.obj
|
|
|
|
# return type of the java method
|
|
r = self.definition_return[0]
|
|
|
|
# now call the java method
|
|
if r == 'V':
|
|
with nogil:
|
|
j_env[0].CallVoidMethodA(
|
|
j_env, j_self, self.j_method, j_args)
|
|
elif r == 'Z':
|
|
with nogil:
|
|
j_boolean = j_env[0].CallBooleanMethodA(
|
|
j_env, j_self, self.j_method, j_args)
|
|
ret = True if j_boolean else False
|
|
elif r == 'B':
|
|
with nogil:
|
|
j_byte = j_env[0].CallByteMethodA(
|
|
j_env, j_self, self.j_method, j_args)
|
|
ret = <char>j_byte
|
|
elif r == 'C':
|
|
with nogil:
|
|
j_char = j_env[0].CallCharMethodA(
|
|
j_env, j_self, self.j_method, j_args)
|
|
ret = chr(<char>j_char)
|
|
elif r == 'S':
|
|
with nogil:
|
|
j_short = j_env[0].CallShortMethodA(
|
|
j_env, j_self, self.j_method, j_args)
|
|
ret = <short>j_short
|
|
elif r == 'I':
|
|
with nogil:
|
|
j_int = j_env[0].CallIntMethodA(
|
|
j_env, j_self, self.j_method, j_args)
|
|
ret = <int>j_int
|
|
elif r == 'J':
|
|
with nogil:
|
|
j_long = j_env[0].CallLongMethodA(
|
|
j_env, j_self, self.j_method, j_args)
|
|
ret = <long long>j_long
|
|
elif r == 'F':
|
|
with nogil:
|
|
j_float = j_env[0].CallFloatMethodA(
|
|
j_env, j_self, self.j_method, j_args)
|
|
ret = <float>j_float
|
|
elif r == 'D':
|
|
with nogil:
|
|
j_double = j_env[0].CallDoubleMethodA(
|
|
j_env, j_self, self.j_method, j_args)
|
|
ret = <double>j_double
|
|
elif r == 'L':
|
|
with nogil:
|
|
j_object = j_env[0].CallObjectMethodA(
|
|
j_env, j_self, self.j_method, j_args)
|
|
check_exception(j_env)
|
|
if j_object != NULL:
|
|
ret = convert_jobject_to_python(
|
|
j_env, self.definition_return, j_object)
|
|
j_env[0].DeleteLocalRef(j_env, j_object)
|
|
elif r == '[':
|
|
r = self.definition_return[1:]
|
|
with nogil:
|
|
j_object = j_env[0].CallObjectMethodA(
|
|
j_env, j_self, self.j_method, j_args)
|
|
check_exception(j_env)
|
|
if j_object != NULL:
|
|
ret = convert_jarray_to_python(j_env, r, j_object)
|
|
j_env[0].DeleteLocalRef(j_env, j_object)
|
|
else:
|
|
raise Exception("Invalid return definition '{}'".format(r))
|
|
|
|
check_exception(j_env)
|
|
return ret
|
|
|
|
cdef call_staticmethod(self, JNIEnv *j_env, jvalue *j_args):
|
|
cdef jboolean j_boolean
|
|
cdef jbyte j_byte
|
|
cdef jchar j_char
|
|
cdef jshort j_short
|
|
cdef jint j_int
|
|
cdef jlong j_long
|
|
cdef jfloat j_float
|
|
cdef jdouble j_double
|
|
cdef jobject j_object
|
|
cdef char *c_str
|
|
cdef bytes py_str
|
|
cdef object ret = None
|
|
cdef JavaObject ret_jobject
|
|
cdef JavaClass ret_jc
|
|
|
|
# return type of the java method
|
|
r = self.definition_return[0]
|
|
|
|
# now call the java method
|
|
if r == 'V':
|
|
with nogil:
|
|
j_env[0].CallStaticVoidMethodA(
|
|
j_env, self.j_cls, self.j_method, j_args)
|
|
elif r == 'Z':
|
|
with nogil:
|
|
j_boolean = j_env[0].CallStaticBooleanMethodA(
|
|
j_env, self.j_cls, self.j_method, j_args)
|
|
ret = True if j_boolean else False
|
|
elif r == 'B':
|
|
with nogil:
|
|
j_byte = j_env[0].CallStaticByteMethodA(
|
|
j_env, self.j_cls, self.j_method, j_args)
|
|
ret = <char>j_byte
|
|
elif r == 'C':
|
|
with nogil:
|
|
j_char = j_env[0].CallStaticCharMethodA(
|
|
j_env, self.j_cls, self.j_method, j_args)
|
|
ret = chr(<char>j_char)
|
|
elif r == 'S':
|
|
with nogil:
|
|
j_short = j_env[0].CallStaticShortMethodA(
|
|
j_env, self.j_cls, self.j_method, j_args)
|
|
ret = <short>j_short
|
|
elif r == 'I':
|
|
with nogil:
|
|
j_int = j_env[0].CallStaticIntMethodA(
|
|
j_env, self.j_cls, self.j_method, j_args)
|
|
ret = <int>j_int
|
|
elif r == 'J':
|
|
with nogil:
|
|
j_long = j_env[0].CallStaticLongMethodA(
|
|
j_env, self.j_cls, self.j_method, j_args)
|
|
ret = <long long>j_long
|
|
elif r == 'F':
|
|
with nogil:
|
|
j_float = j_env[0].CallStaticFloatMethodA(
|
|
j_env, self.j_cls, self.j_method, j_args)
|
|
ret = <float>j_float
|
|
elif r == 'D':
|
|
with nogil:
|
|
j_double = j_env[0].CallStaticDoubleMethodA(
|
|
j_env, self.j_cls, self.j_method, j_args)
|
|
ret = <double>j_double
|
|
elif r == 'L':
|
|
with nogil:
|
|
j_object = j_env[0].CallStaticObjectMethodA(
|
|
j_env, self.j_cls, self.j_method, j_args)
|
|
check_exception(j_env)
|
|
if j_object != NULL:
|
|
ret = convert_jobject_to_python(
|
|
j_env, self.definition_return, j_object)
|
|
j_env[0].DeleteLocalRef(j_env, j_object)
|
|
elif r == '[':
|
|
r = self.definition_return[1:]
|
|
with nogil:
|
|
j_object = j_env[0].CallStaticObjectMethodA(
|
|
j_env, self.j_cls, self.j_method, j_args)
|
|
check_exception(j_env)
|
|
if j_object != NULL:
|
|
ret = convert_jarray_to_python(j_env, r, j_object)
|
|
j_env[0].DeleteLocalRef(j_env, j_object)
|
|
else:
|
|
raise Exception("Invalid return definition '{}'".format(r))
|
|
|
|
check_exception(j_env)
|
|
return ret
|
|
|
|
|
|
cdef class JavaMultipleMethod(object):
|
|
|
|
cdef LocalRef j_self
|
|
cdef list definitions
|
|
cdef dict static_methods
|
|
cdef dict instance_methods
|
|
cdef bytes name
|
|
cdef bytes classname
|
|
|
|
def __cinit__(self, definition, **kwargs):
|
|
self.j_self = None
|
|
|
|
def __init__(self, definitions, **kwargs):
|
|
super(JavaMultipleMethod, self).__init__()
|
|
self.definitions = definitions
|
|
self.static_methods = {}
|
|
self.instance_methods = {}
|
|
self.name = None
|
|
|
|
def __get__(self, obj, objtype):
|
|
if obj is None:
|
|
self.j_self = None
|
|
return self
|
|
# XXX FIXME we MUST not change our own j_self, but return a "bound"
|
|
# method here, as python does!
|
|
cdef JavaClass jc = obj
|
|
self.j_self = jc.j_self
|
|
return self
|
|
|
|
cdef void set_resolve_info(self, JNIEnv *j_env, jclass j_cls,
|
|
LocalRef j_self, bytes name, bytes classname):
|
|
cdef JavaMethod jm
|
|
self.name = name
|
|
self.classname = classname
|
|
|
|
for signature, static, is_varargs in self.definitions:
|
|
jm = None
|
|
if j_self is None and static:
|
|
if signature in self.static_methods:
|
|
continue
|
|
jm = JavaStaticMethod(signature, varargs=is_varargs)
|
|
jm.set_resolve_info(j_env, j_cls, j_self, name, classname)
|
|
self.static_methods[signature] = jm
|
|
|
|
elif j_self is not None and not static:
|
|
if signature in self.instance_methods:
|
|
continue
|
|
jm = JavaMethod(signature, varargs=is_varargs)
|
|
jm.set_resolve_info(j_env, j_cls, None, name, classname)
|
|
self.instance_methods[signature] = jm
|
|
|
|
def __call__(self, *args):
|
|
# try to match our args to a signature
|
|
cdef JavaMethod jm
|
|
cdef list scores = []
|
|
cdef dict methods
|
|
cdef int max_sign_args
|
|
cdef list found_signatures = []
|
|
|
|
if self.j_self:
|
|
methods = self.instance_methods
|
|
else:
|
|
methods = self.static_methods
|
|
|
|
for signature, jm in items_compat(methods):
|
|
# store signatures for the exception
|
|
found_signatures.append(signature)
|
|
|
|
sign_ret, sign_args = jm.definition_return, jm.definition_args
|
|
if jm.is_varargs:
|
|
max_sign_args = len(sign_args) - 1
|
|
args_ = args[:max_sign_args] + (args[max_sign_args:],)
|
|
else:
|
|
args_ = args
|
|
|
|
score = calculate_score(sign_args, args_, jm.is_varargs)
|
|
|
|
if score <= 0:
|
|
continue
|
|
scores.append((score, signature))
|
|
|
|
if not scores:
|
|
raise JavaException(
|
|
'No methods matching your arguments, available: {}'.format(
|
|
found_signatures
|
|
)
|
|
)
|
|
scores.sort()
|
|
score, signature = scores[-1]
|
|
|
|
jm = methods[signature]
|
|
jm.j_self = self.j_self
|
|
return jm.__call__(*args)
|
|
|
|
|
|
class JavaStaticMethod(JavaMethod):
|
|
def __init__(self, definition, **kwargs):
|
|
kwargs['static'] = True
|
|
super(JavaStaticMethod, self).__init__(definition, **kwargs)
|
|
|
|
|
|
class JavaStaticField(JavaField):
|
|
def __init__(self, definition, **kwargs):
|
|
kwargs['static'] = True
|
|
super(JavaStaticField, self).__init__(definition, **kwargs)
|