package org.jruby.java.addons;
import org.jruby.Ruby;
import org.jruby.RubyClass;
import org.jruby.RubyString;
import org.jruby.anno.JRubyMethod;
import org.jruby.java.proxies.JavaProxy;
import org.jruby.javasupport.Java;
import org.jruby.javasupport.JavaClass;
import org.jruby.runtime.ThreadContext;
import org.jruby.runtime.builtin.IRubyObject;
import java.lang.reflect.Field;
import java.util.Collection;
import java.util.Set;
public abstract class ClassJavaAddons {
@JRubyMethod
public static IRubyObject java_class(ThreadContext context, final IRubyObject self) {
Class reifiedClass = RubyClass.nearestReifiedClass((RubyClass) self);
if ( reifiedClass == null ) return context.nil;
return asJavaClass(context.runtime, reifiedClass);
}
private static IRubyObject asJavaClass(final Ruby runtime, final Class<?> reifiedClass) {
return Java.getInstance(runtime, reifiedClass);
}
@JRubyMethod(name = "become_java!", required = 0)
public static IRubyObject become_java(ThreadContext context, final IRubyObject self) {
return becomeJava(context, (RubyClass) self, null, true);
}
@JRubyMethod(name = "become_java!", required = 1, optional = 1)
public static IRubyObject become_java(ThreadContext context, final IRubyObject self, final IRubyObject[] args) {
final RubyClass klass = (RubyClass) self;
String dumpDir = null; boolean useChildLoader = true;
if ( args[0] instanceof RubyString ) {
dumpDir = args[0].asJavaString();
if ( args.length > 1 ) useChildLoader = args[1].isTrue();
}
else {
useChildLoader = args[0].isTrue();
if ( args.length > 1 && args[1] instanceof RubyString ) {
dumpDir = args[1].asJavaString();
}
}
return becomeJava(context, klass, dumpDir, useChildLoader);
}
private static IRubyObject becomeJava(final ThreadContext context, final RubyClass klass,
final String dumpDir, final boolean useChildLoader) {
klass.reifyWithAncestors(dumpDir, useChildLoader);
final Class<?> reifiedClass = klass.getReifiedClass();
generateFieldAccessors(context, klass, reifiedClass);
return asJavaClass(context.runtime, reifiedClass);
}
private static void generateFieldAccessors(ThreadContext context, final RubyClass klass, final Class<?> javaClass) {
for ( String name : getJavaFieldNames(klass) ) {
Field field;
try {
field = javaClass.getDeclaredField(name);
}
catch (NoSuchFieldException e) {
throw context.runtime.newRuntimeError("no field: '" + name + "' in reified class for " + klass.getName());
}
JavaProxy.installField(context, name, field, klass);
}
}
private static Set<String> getJavaFieldNames(final RubyClass klass) {
return klass.getFieldSignatures().keySet();
}
}