org.jruby.runtime.callback.ReflectionCallback Maven / Gradle / Ivy
/***** BEGIN LICENSE BLOCK *****
* Version: CPL 1.0/GPL 2.0/LGPL 2.1
*
* The contents of this file are subject to the Common 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/cpl-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) 2002-2004 Anders Bengtsson
* Copyright (C) 2004 Jan Arne Petersen
* Copyright (C) 2004-2005 Thomas E Enebo
* Copyright (C) 2004 Stefan Matthias Aust
* Copyright (C) 2005 Charles O Nutter
*
* 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 CPL, 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 CPL, the GPL or the LGPL.
***** END LICENSE BLOCK *****/
package org.jruby.runtime.callback;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import org.jruby.exceptions.JumpException;
import org.jruby.exceptions.MainExitException;
import org.jruby.exceptions.RaiseException;
import org.jruby.exceptions.ThreadKill;
import org.jruby.runtime.Arity;
import org.jruby.runtime.Block;
import org.jruby.runtime.builtin.IRubyObject;
/**
* A wrapper for java.lang.reflect.Method
objects which implement Ruby methods.
*/
public class ReflectionCallback implements Callback {
private Method method;
private Class type;
private String methodName;
private Class[] argumentTypes;
private boolean isRestArgs;
private Arity arity;
private boolean isStaticMethod;
private boolean fast;
public ReflectionCallback(Class type, String methodName, Class[] argumentTypes,
boolean isRestArgs, boolean isStaticMethod, Arity arity, boolean fast) {
this.type = type;
this.methodName = methodName;
this.argumentTypes = argumentTypes;
this.isRestArgs = isRestArgs;
this.isStaticMethod = isStaticMethod;
this.arity = arity;
this.fast = fast;
assert type != null;
assert methodName != null;
assert arity != null;
loadMethod(fast);
}
private void loadMethod(boolean fast) {
Class[] args;
if (isStaticMethod) {
Class[] types = new Class[argumentTypes.length + 1];
System.arraycopy(argumentTypes, 0, types, 1, argumentTypes.length);
types[0] = IRubyObject.class;
args = types;
} else {
args = argumentTypes;
}
// ENEBO: Perhaps slow but simple for now
if (!fast) {
Class[] types = new Class[args.length + 1];
System.arraycopy(args, 0, types, 0, args.length);
types[args.length] = Block.class;
args = types;
}
try {
method = type.getMethod(methodName, args);
} catch (NoSuchMethodException e) {
throw new RuntimeException("NoSuchMethodException: Cannot get method \"" + methodName
+ "\" in class \"" + type.getName() + "\" by Reflection.");
} catch (SecurityException e) {
throw new RuntimeException("SecurityException: Cannot get method \"" + methodName
+ "\" in class \"" + type.getName() + "\" by Reflection.");
}
}
/**
* Returns an object array that collects all rest arguments in its own object array which
* is then put into the last slot of the first object array. That is, assuming that this
* callback expects one required argument and any number of rest arguments, an input of
* [1, 2, 3]
is transformed into [1, [2, 3]]
.
*/
protected final Object[] packageRestArgumentsForReflection(final Object[] originalArgs) {
IRubyObject[] restArray = new IRubyObject[originalArgs.length - (argumentTypes.length - 1)];
Object[] result = new Object[argumentTypes.length];
try {
System.arraycopy(originalArgs, argumentTypes.length - 1, restArray, 0, originalArgs.length - (argumentTypes.length - 1));
} catch (ArrayIndexOutOfBoundsException e) {
assert false : e;
return null;
}
System.arraycopy(originalArgs, 0, result, 0, argumentTypes.length - 1);
result[argumentTypes.length - 1] = restArray;
return result;
}
/**
* Invokes the Ruby method. Actually, this methods delegates to an internal version
* that may throw the usual Java reflection exceptions. Ruby exceptions are rethrown,
* other exceptions throw an AssertError and abort the execution of the Ruby program.
* They should never happen.
*/
/**
* Calls a wrapped Ruby method for the specified receiver with the specified arguments.
*/
public IRubyObject execute(IRubyObject recv, IRubyObject[] oargs, Block block) {
arity.checkArity(recv.getRuntime(), oargs);
Object[] methodArgs = oargs;
if (isRestArgs) {
methodArgs = packageRestArgumentsForReflection(methodArgs);
}
try {
IRubyObject receiver = recv;
if (isStaticMethod) {
Object[] args = new Object[methodArgs.length + (fast ? 1 : 2)];
System.arraycopy(methodArgs, 0, args, 1, methodArgs.length);
args[0] = recv;
if (!fast) args[methodArgs.length + 1] = block;
receiver = null;
methodArgs = args;
} else {
Object[] args = new Object[methodArgs.length + (fast ? 0 : 1)];
System.arraycopy(methodArgs, 0, args, 0, methodArgs.length);
if (!fast) args[methodArgs.length] = block;
methodArgs = args;
}
return (IRubyObject) method.invoke(receiver, methodArgs);
} catch (InvocationTargetException e) {
if (e.getTargetException() instanceof RaiseException) {
throw (RaiseException) e.getTargetException();
} else if (e.getTargetException() instanceof JumpException) {
throw (JumpException) e.getTargetException();
} else if (e.getTargetException() instanceof ThreadKill) {
// allow it to bubble up
throw (ThreadKill) e.getTargetException();
} else if (e.getTargetException() instanceof Exception) {
if(e.getTargetException() instanceof MainExitException) {
throw (RuntimeException)e.getTargetException();
}
recv.getRuntime().getJavaSupport().handleNativeException(e.getTargetException(), method);
return recv.getRuntime().getNil();
} else {
throw (Error) e.getTargetException();
}
} catch (IllegalAccessException e) {
StringBuilder message = new StringBuilder();
message.append(e.getMessage());
message.append(':');
message.append(" methodName=").append(methodName);
message.append(" recv=").append(recv.toString());
message.append(" type=").append(type.getName());
message.append(" methodArgs=[");
for (int i = 0; i < methodArgs.length; i++) {
message.append(methodArgs[i]);
message.append(' ');
}
message.append(']');
assert false : message.toString();
return null;
} catch (final IllegalArgumentException e) {
/* StringBuilder message = new StringBuilder();
message.append(e.getMessage());
message.append(':');
message.append(" methodName=").append(methodName);
message.append(" recv=").append(recv.toString());
message.append(" type=").append(type.getName());
message.append(" methodArgs=[");
for (int i = 0; i < methodArgs.length; i++) {
message.append(methodArgs[i]);
message.append(' ');
}
message.append(']');*/
assert false : e;
return null;
}
}
/**
* Returns the arity of the wrapped Ruby method.
*/
public Arity getArity() {
return arity;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy