org.jruby.javasupport.JavaMethod Maven / Gradle / Ivy
/***** BEGIN LICENSE BLOCK *****
* Version: EPL 1.0/GPL 2.0/LGPL 2.1
*
* The contents of this file are subject to the Eclipse Public
* License Version 1.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.eclipse.org/legal/epl-v10.html
*
* Software distributed under the License is distributed on an "AS
* IS" basis, WITHOUT WARRANTY OF ANY KIND, either express or
* implied. See the License for the specific language governing
* rights and limitations under the License.
*
* Copyright (C) 2001 Chad Fowler
* Copyright (C) 2001 Alan Moore
* Copyright (C) 2001-2004 Jan Arne Petersen
* Copyright (C) 2002 Benoit Cerrina
* Copyright (C) 2002-2004 Anders Bengtsson
* Copyright (C) 2004 Stefan Matthias Aust
* Copyright (C) 2004 Thomas E Enebo
* Copyright (C) 2004 David Corbin
* Copyright (C) 2005 Charles O Nutter
* Copyright (C) 2006 Kresten Krab Thorup
*
* Alternatively, the contents of this file may be used under the terms of
* either of the GNU General Public License Version 2 or later (the "GPL"),
* or the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
* in which case the provisions of the GPL or the LGPL are applicable instead
* of those above. If you wish to allow use of your version of this file only
* under the terms of either the GPL or the LGPL, and not to allow others to
* use your version of this file under the terms of the EPL, indicate your
* decision by deleting the provisions above and replace them with the notice
* and other provisions required by the GPL or the LGPL. If you do not delete
* the provisions above, a recipient may use your version of this file under
* the terms of any one of the EPL, the GPL or the LGPL.
***** END LICENSE BLOCK *****/
package org.jruby.javasupport;
import java.lang.annotation.Annotation;
import java.lang.reflect.AccessibleObject;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
import java.lang.reflect.Type;
import org.jruby.Ruby;
import org.jruby.RubyBoolean;
import org.jruby.RubyClass;
import org.jruby.RubyModule;
import org.jruby.RubyString;
import org.jruby.anno.JRubyClass;
import org.jruby.anno.JRubyMethod;
import org.jruby.exceptions.RaiseException;
import org.jruby.javasupport.proxy.InternalJavaProxy;
import org.jruby.javasupport.proxy.JavaProxyClass;
import org.jruby.javasupport.proxy.JavaProxyMethod;
import org.jruby.runtime.ObjectAllocator;
import org.jruby.runtime.builtin.IRubyObject;
import static org.jruby.util.CodegenUtils.getBoxType;
import static org.jruby.util.CodegenUtils.prettyParams;
@JRubyClass(name="Java::JavaMethod")
public class JavaMethod extends JavaCallable {
//private final static boolean USE_HANDLES = RubyInstanceConfig.USE_GENERATED_HANDLES;
//private final static boolean HANDLE_DEBUG = false;
private final Method method;
private final Class> boxedReturnType;
private final boolean isFinal;
private final JavaUtil.JavaConverter returnConverter;
public final Method getValue() { return method; }
public static RubyClass createJavaMethodClass(Ruby runtime, RubyModule javaModule) {
// TODO: NOT_ALLOCATABLE_ALLOCATOR is probably ok here, since we don't intend for people to monkey with
// this type and it can't be marshalled. Confirm. JRUBY-415
RubyClass result =
javaModule.defineClassUnder("JavaMethod", runtime.getObject(), ObjectAllocator.NOT_ALLOCATABLE_ALLOCATOR);
JavaAccessibleObject.registerRubyMethods(runtime, result);
JavaCallable.registerRubyMethods(runtime, result);
result.defineAnnotatedMethods(JavaMethod.class);
return result;
}
public JavaMethod(Ruby runtime, Method method) {
super(runtime, runtime.getJavaSupport().getJavaMethodClass(), method.getParameterTypes());
this.method = method;
this.isFinal = Modifier.isFinal(method.getModifiers());
final Class> returnType = method.getReturnType();
if (returnType.isPrimitive() && returnType != void.class) {
this.boxedReturnType = getBoxType(returnType);
} else {
this.boxedReturnType = returnType;
}
// Special classes like Collections.EMPTY_LIST are inner classes that are private but
// implement public interfaces. Their methods are all public methods for the public
// interface. Let these public methods execute via setAccessible(true).
if (JavaUtil.CAN_SET_ACCESSIBLE) {
// we should be able to setAccessible ok...
try {
if ( Modifier.isPublic(method.getModifiers()) &&
! Modifier.isPublic(method.getDeclaringClass().getModifiers()) ) {
method.setAccessible(true);
}
} catch (SecurityException se) {
// we shouldn't get here if JavaClass.CAN_SET_ACCESSIBLE is doing
// what it should, so we warn.
runtime.getWarnings().warn("failed to setAccessible: " + method + ", exception follows: " + se.getMessage());
}
}
returnConverter = JavaUtil.getJavaConverter(returnType);
}
@Deprecated // no-longer used
public static JavaMethod create(Ruby runtime, Method method) {
return new JavaMethod(runtime, method);
}
@Deprecated // no-longer used
public static JavaMethod create(Ruby runtime, Class> javaClass, String methodName, Class>[] argumentTypes) {
try {
return create(runtime, javaClass.getMethod(methodName, argumentTypes));
}
catch (NoSuchMethodException e) {
throw runtime.newNameError("undefined method '" + methodName + "' for class '" + javaClass.getName() + "'", methodName);
}
}
@Deprecated // no-longer used
public static JavaMethod createDeclared(Ruby runtime, Class> javaClass, String methodName, Class>[] argumentTypes) {
try {
return create(runtime, javaClass.getDeclaredMethod(methodName, argumentTypes));
}
catch (NoSuchMethodException e) {
throw runtime.newNameError("undefined method '" + methodName + "' for class '" + javaClass.getName() + "'", methodName);
}
}
public static JavaMethod getMatchingDeclaredMethod(Ruby runtime, Class> javaClass, String methodName, Class>[] argumentTypes) {
// FIXME: do we really want 'declared' methods? includes private/protected, and does _not_
// include superclass methods. also, the getDeclared calls may throw SecurityException if
// we're running under a restrictive security policy.
try {
return new JavaMethod(runtime, javaClass.getDeclaredMethod(methodName, argumentTypes));
}
catch (NoSuchMethodException e) {
// search through all declared methods to find a closest match
MethodSearch: for ( Method method : javaClass.getDeclaredMethods() ) {
if ( method.getName().equals(methodName) ) {
Class>[] targetTypes = method.getParameterTypes();
// for zero args case we can stop searching
if (targetTypes.length == 0 && argumentTypes.length == 0) {
return new JavaMethod(runtime, method);
}
TypeScan: for (int i = 0; i < argumentTypes.length; i++) {
if (i >= targetTypes.length) continue MethodSearch;
if (targetTypes[i].isAssignableFrom(argumentTypes[i])) {
continue TypeScan;
} else {
continue MethodSearch;
}
}
// if we get here, we found a matching method, use it
// TODO: choose narrowest method by continuing to search
return new JavaMethod(runtime, method);
}
}
}
// no matching method found
return null;
}
@Override
public final boolean equals(Object other) {
return other instanceof JavaMethod && this.method.equals( ((JavaMethod) other).method );
}
@Override
public final int hashCode() {
return method.hashCode();
}
@JRubyMethod
@Override
public RubyString name() {
return getRuntime().newString(method.getName());
}
@JRubyMethod(name = "public?")
@Override
public RubyBoolean public_p() {
return getRuntime().newBoolean(Modifier.isPublic(method.getModifiers()));
}
@JRubyMethod(name = "final?")
public RubyBoolean final_p() {
return getRuntime().newBoolean(Modifier.isFinal(method.getModifiers()));
}
@JRubyMethod(rest = true)
public IRubyObject invoke(IRubyObject[] args) {
checkArity(args.length - 1);
final IRubyObject invokee = args[0];
final Object[] arguments = convertArguments(args, 1);
if ( invokee.isNil() ) {
return invokeWithExceptionHandling(method, null, arguments);
}
final Object javaInvokee;
if (!isStatic()) {
javaInvokee = JavaUtil.unwrapJavaValue(invokee);
if ( javaInvokee == null ) {
throw getRuntime().newTypeError("invokee not a java object");
}
if ( ! method.getDeclaringClass().isInstance(javaInvokee) ) {
throw getRuntime().newTypeError(
"invokee not instance of method's class" +
" (got" + javaInvokee.getClass().getName() +
" wanted " + method.getDeclaringClass().getName() + ")");
}
//
// this test really means, that this is a ruby-defined subclass of a java class
//
if ( javaInvokee instanceof InternalJavaProxy &&
// don't bother to check if final method, it won't
// be there (not generated, can't be!)
! isFinal ) {
JavaProxyClass jpc = ((InternalJavaProxy) javaInvokee).___getProxyClass();
JavaProxyMethod jpm = jpc.getMethod( method.getName(), parameterTypes );
if ( jpm != null && jpm.hasSuperImplementation() ) {
return invokeWithExceptionHandling(jpm.getSuperMethod(), javaInvokee, arguments);
}
}
}
else {
javaInvokee = null;
}
return invokeWithExceptionHandling(method, javaInvokee, arguments);
}
@JRubyMethod(rest = true)
public IRubyObject invoke_static(IRubyObject[] args) {
checkArity(args.length);
final Object[] arguments = convertArguments(args, 0);
return invokeWithExceptionHandling(method, null, arguments);
}
@JRubyMethod
public IRubyObject return_type() {
Class> klass = method.getReturnType();
if (klass.equals(void.class)) {
return getRuntime().getNil();
}
return JavaClass.get(getRuntime(), klass);
}
@JRubyMethod
public IRubyObject type_parameters() {
return Java.getInstance(getRuntime(), method.getTypeParameters());
}
public IRubyObject invokeDirect(Object javaInvokee, Object[] args) {
checkArity(args.length);
checkInstanceof(javaInvokee);
if (mightBeProxy(javaInvokee)) {
return tryProxyInvocation(javaInvokee, args);
}
return invokeDirectWithExceptionHandling(method, javaInvokee, args);
}
public IRubyObject invokeDirect(Object javaInvokee) {
assert method.getDeclaringClass().isInstance(javaInvokee);
checkArity(0);
if (mightBeProxy(javaInvokee)) {
return tryProxyInvocation(javaInvokee);
}
return invokeDirectWithExceptionHandling(method, javaInvokee);
}
public IRubyObject invokeDirect(Object javaInvokee, Object arg0) {
assert method.getDeclaringClass().isInstance(javaInvokee);
checkArity(1);
if (mightBeProxy(javaInvokee)) {
return tryProxyInvocation(javaInvokee, arg0);
}
return invokeDirectWithExceptionHandling(method, javaInvokee, arg0);
}
public IRubyObject invokeDirect(Object javaInvokee, Object arg0, Object arg1) {
assert method.getDeclaringClass().isInstance(javaInvokee);
checkArity(2);
if (mightBeProxy(javaInvokee)) {
return tryProxyInvocation(javaInvokee, arg0, arg1);
}
return invokeDirectWithExceptionHandling(method, javaInvokee, arg0, arg1);
}
public IRubyObject invokeDirect(Object javaInvokee, Object arg0, Object arg1, Object arg2) {
assert method.getDeclaringClass().isInstance(javaInvokee);
checkArity(3);
if (mightBeProxy(javaInvokee)) {
return tryProxyInvocation(javaInvokee, arg0, arg1, arg2);
}
return invokeDirectWithExceptionHandling(method, javaInvokee, arg0, arg1, arg2);
}
public IRubyObject invokeDirect(Object javaInvokee, Object arg0, Object arg1, Object arg2, Object arg3) {
assert method.getDeclaringClass().isInstance(javaInvokee);
checkArity(4);
if (mightBeProxy(javaInvokee)) {
return tryProxyInvocation(javaInvokee, arg0, arg1, arg2, arg3);
}
return invokeDirectWithExceptionHandling(method, javaInvokee, arg0, arg1, arg2, arg3);
}
public IRubyObject invokeStaticDirect(Object[] args) {
checkArity(args.length);
return invokeDirectWithExceptionHandling(method, null, args);
}
public IRubyObject invokeStaticDirect() {
checkArity(0);
return invokeDirectWithExceptionHandling(method, null);
}
public IRubyObject invokeStaticDirect(Object arg0) {
checkArity(1);
return invokeDirectWithExceptionHandling(method, null, arg0);
}
public IRubyObject invokeStaticDirect(Object arg0, Object arg1) {
checkArity(2);
return invokeDirectWithExceptionHandling(method, null, arg0, arg1);
}
public IRubyObject invokeStaticDirect(Object arg0, Object arg1, Object arg2) {
checkArity(3);
return invokeDirectWithExceptionHandling(method, null, arg0, arg1, arg2);
}
public IRubyObject invokeStaticDirect(Object arg0, Object arg1, Object arg2, Object arg3) {
checkArity(4);
return invokeDirectWithExceptionHandling(method, null, arg0, arg1, arg2, arg3);
}
private void checkInstanceof(Object javaInvokee) throws RaiseException {
if (!method.getDeclaringClass().isInstance(javaInvokee)) {
throw getRuntime().newTypeError("invokee not instance of method's class (" + "got" + javaInvokee.getClass().getName() + " wanted " + method.getDeclaringClass().getName() + ")");
}
}
private IRubyObject invokeWithExceptionHandling(Method method, Object javaInvokee, Object[] arguments) {
try {
Object result = method.invoke(javaInvokee, arguments);
return returnConverter.convert(getRuntime(), result);
} catch (IllegalArgumentException iae) {
return handlelIllegalArgumentEx(iae, method, arguments);
} catch (IllegalAccessException iae) {
return handleIllegalAccessEx(iae, method);
} catch (InvocationTargetException ite) {
return handleInvocationTargetEx(ite, method);
} catch (Throwable t) {
return handleThrowable(t, method);
}
}
private IRubyObject invokeDirectSuperWithExceptionHandling(Method method, Object javaInvokee, Object... arguments) {
// super calls from proxies must use reflected method
// FIXME: possible to make handles do the superclass call?
try {
Object result = method.invoke(javaInvokee, arguments);
return convertReturn(result);
} catch (IllegalArgumentException iae) {
return handlelIllegalArgumentEx(iae, method, arguments);
} catch (IllegalAccessException iae) {
return handleIllegalAccessEx(iae, method);
} catch (InvocationTargetException ite) {
return handleInvocationTargetEx(ite, method);
} catch (Throwable t) {
return handleThrowable(t, method);
}
}
private IRubyObject invokeDirectWithExceptionHandling(Method method, Object javaInvokee, Object[] arguments) {
try {
Object result = method.invoke(javaInvokee, arguments);
return convertReturn(result);
} catch (IllegalArgumentException iae) {
return handlelIllegalArgumentEx(iae, method, arguments);
} catch (IllegalAccessException iae) {
return handleIllegalAccessEx(iae, method);
} catch (InvocationTargetException ite) {
return handleInvocationTargetEx(ite, method);
} catch (Throwable t) {
return handleThrowable(t, method);
}
}
private IRubyObject invokeDirectWithExceptionHandling(Method method, Object javaInvokee) {
try {
Object result = method.invoke(javaInvokee);
return convertReturn(result);
} catch (IllegalArgumentException iae) {
return handlelIllegalArgumentEx(iae, method);
} catch (IllegalAccessException iae) {
return handleIllegalAccessEx(iae, method);
} catch (InvocationTargetException ite) {
return handleInvocationTargetEx(ite, method);
} catch (Throwable t) {
return handleThrowable(t, method);
}
}
private IRubyObject invokeDirectWithExceptionHandling(Method method, Object javaInvokee, Object arg0) {
try {
Object result = method.invoke(javaInvokee, arg0);
return convertReturn(result);
} catch (IllegalArgumentException iae) {
return handlelIllegalArgumentEx(iae, method, arg0);
} catch (IllegalAccessException iae) {
return handleIllegalAccessEx(iae, method);
} catch (InvocationTargetException ite) {
return handleInvocationTargetEx(ite, method);
} catch (Throwable t) {
return handleThrowable(t, method);
}
}
private IRubyObject invokeDirectWithExceptionHandling(Method method, Object javaInvokee, Object arg0, Object arg1) {
try {
Object result = method.invoke(javaInvokee, arg0, arg1);
return convertReturn(result);
} catch (IllegalArgumentException iae) {
return handlelIllegalArgumentEx(iae, method, arg0, arg1);
} catch (IllegalAccessException iae) {
return handleIllegalAccessEx(iae, method);
} catch (InvocationTargetException ite) {
return handleInvocationTargetEx(ite, method);
} catch (Throwable t) {
return handleThrowable(t, method);
}
}
private IRubyObject invokeDirectWithExceptionHandling(Method method, Object javaInvokee, Object arg0, Object arg1, Object arg2) {
try {
Object result = method.invoke(javaInvokee, arg0, arg1, arg2);
return convertReturn(result);
} catch (IllegalArgumentException iae) {
return handlelIllegalArgumentEx(iae, method, arg0, arg1, arg2);
} catch (IllegalAccessException iae) {
return handleIllegalAccessEx(iae, method);
} catch (InvocationTargetException ite) {
return handleInvocationTargetEx(ite, method);
} catch (Throwable t) {
return handleThrowable(t, method);
}
}
private IRubyObject invokeDirectWithExceptionHandling(Method method, Object javaInvokee, Object arg0, Object arg1, Object arg2, Object arg3) {
try {
Object result = method.invoke(javaInvokee, arg0, arg1, arg2, arg3);
return convertReturn(result);
} catch (IllegalArgumentException iae) {
return handlelIllegalArgumentEx(iae, method, arg0, arg1, arg2, arg3);
} catch (IllegalAccessException iae) {
return handleIllegalAccessEx(iae, method);
} catch (InvocationTargetException ite) {
return handleInvocationTargetEx(ite, method);
} catch (Throwable t) {
return handleThrowable(t, method);
}
}
private IRubyObject convertReturn(Object result) {
if (result != null && result.getClass() != boxedReturnType) {
// actual type does not exactly match method return type, re-get converter
// FIXME: when the only autoconversions are primitives, this won't be needed
return JavaUtil.convertJavaToUsableRubyObject(getRuntime(), result);
}
return JavaUtil.convertJavaToUsableRubyObjectWithConverter(getRuntime(), result, returnConverter);
}
//@Override
//public final int getArity() {
// return parameterTypes.length;
//}
//@Override
//public final Class>[] getParameterTypes() {
// return parameterTypes;
//}
@Override
public Class>[] getExceptionTypes() {
return method.getExceptionTypes();
}
@Override
public Type[] getGenericParameterTypes() {
return method.getGenericParameterTypes();
}
@Override
public Type[] getGenericExceptionTypes() {
return method.getGenericExceptionTypes();
}
@Override
public Annotation[][] getParameterAnnotations() {
return method.getParameterAnnotations();
}
@Override
public final boolean isVarArgs() {
return method.isVarArgs();
}
@Override // not used
protected String nameOnInspection() {
return getType().toString() + '/' + method.getName();
}
@JRubyMethod
public RubyString inspect() {
StringBuilder str = new StringBuilder();
str.append("#<");
str.append( getType().toString() ).append('/').append(method.getName());
inspectParameterTypes(str, this);
str.append(">");
return getRuntime().newString( str.toString() );
}
@JRubyMethod(name = "static?")
public RubyBoolean static_p() {
return getRuntime().newBoolean(isStatic());
}
public RubyBoolean bridge_p() {
return getRuntime().newBoolean(method.isBridge());
}
private boolean isStatic() {
return Modifier.isStatic(method.getModifiers());
}
@Override
public final int getModifiers() {
return method.getModifiers();
}
@Override
public String toGenericString() {
return method.toGenericString();
}
@Override
public final AccessibleObject accessibleObject() {
return method;
}
private boolean mightBeProxy(Object javaInvokee) {
// this test really means, that this is a ruby-defined subclass of a java class
return javaInvokee instanceof InternalJavaProxy && !isFinal;
}
private IRubyObject tryProxyInvocation(Object javaInvokee, Object... args) {
JavaProxyClass jpc = ((InternalJavaProxy) javaInvokee).___getProxyClass();
JavaProxyMethod jpm;
if ((jpm = jpc.getMethod(method.getName(), parameterTypes)) != null && jpm.hasSuperImplementation()) {
return invokeDirectSuperWithExceptionHandling(jpm.getSuperMethod(), javaInvokee, args);
} else {
return invokeDirectWithExceptionHandling(method, javaInvokee, args);
}
}
private IRubyObject tryProxyInvocation(Object javaInvokee) {
JavaProxyClass jpc = ((InternalJavaProxy) javaInvokee).___getProxyClass();
JavaProxyMethod jpm;
if ((jpm = jpc.getMethod(method.getName(), parameterTypes)) != null && jpm.hasSuperImplementation()) {
return invokeDirectSuperWithExceptionHandling(jpm.getSuperMethod(), javaInvokee);
} else {
return invokeDirectWithExceptionHandling(method, javaInvokee);
}
}
private IRubyObject tryProxyInvocation(Object javaInvokee, Object arg0) {
JavaProxyClass jpc = ((InternalJavaProxy) javaInvokee).___getProxyClass();
JavaProxyMethod jpm;
if ((jpm = jpc.getMethod(method.getName(), parameterTypes)) != null && jpm.hasSuperImplementation()) {
return invokeDirectSuperWithExceptionHandling(jpm.getSuperMethod(), javaInvokee, arg0);
} else {
return invokeDirectWithExceptionHandling(method, javaInvokee, arg0);
}
}
private IRubyObject tryProxyInvocation(Object javaInvokee, Object arg0, Object arg1) {
JavaProxyClass jpc = ((InternalJavaProxy) javaInvokee).___getProxyClass();
JavaProxyMethod jpm;
if ((jpm = jpc.getMethod(method.getName(), parameterTypes)) != null && jpm.hasSuperImplementation()) {
return invokeDirectSuperWithExceptionHandling(jpm.getSuperMethod(), javaInvokee, arg0, arg1);
} else {
return invokeDirectWithExceptionHandling(method, javaInvokee, arg0, arg1);
}
}
private IRubyObject tryProxyInvocation(Object javaInvokee, Object arg0, Object arg1, Object arg2) {
JavaProxyClass jpc = ((InternalJavaProxy) javaInvokee).___getProxyClass();
JavaProxyMethod jpm;
if ((jpm = jpc.getMethod(method.getName(), parameterTypes)) != null && jpm.hasSuperImplementation()) {
return invokeDirectSuperWithExceptionHandling(jpm.getSuperMethod(), javaInvokee, arg0, arg1, arg2);
} else {
return invokeDirectWithExceptionHandling(method, javaInvokee, arg0, arg1, arg2);
}
}
private IRubyObject tryProxyInvocation(Object javaInvokee, Object arg0, Object arg1, Object arg2, Object arg3) {
JavaProxyClass jpc = ((InternalJavaProxy) javaInvokee).___getProxyClass();
JavaProxyMethod jpm;
if ((jpm = jpc.getMethod(method.getName(), parameterTypes)) != null && jpm.hasSuperImplementation()) {
return invokeDirectSuperWithExceptionHandling(jpm.getSuperMethod(), javaInvokee, arg0, arg1, arg2, arg3);
} else {
return invokeDirectWithExceptionHandling(method, javaInvokee, arg0, arg1, arg2, arg3);
}
}
public static RaiseException newMethodNotFoundError(Ruby runtime, Class target, String prettyName, String simpleName) {
return runtime.newNameError("java method not found: " + target.getName() + "." + prettyName, simpleName);
}
public static RaiseException newArgSizeMismatchError(Ruby runtime, Class ... argTypes) {
return runtime.newArgumentError("argument count mismatch for method signature " + prettyParams(argTypes));
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy