checkpoint
[sbp.git] / src / edu / berkeley / sbp / util / Reflection.java
index 327a1dd..3a129d4 100644 (file)
@@ -1,6 +1,7 @@
 package edu.berkeley.sbp.util;
 import java.io.*;
 import java.lang.reflect.*;
+import java.lang.annotation.*;
 
 /** Random reflection-related utilities */
 public final class Reflection {
@@ -202,6 +203,12 @@ public final class Reflection {
     public static interface Show {
     }
 
+    public static Object impose(Object o, Object[] fields) {
+        if (o instanceof Class) return impose((Class)o, fields);
+        if (o instanceof Method) return impose((Method)o, fields);
+        if (o instanceof Constructor) return impose((Constructor)o, fields);
+        return null;
+    }
     public static Object impose(Class _class, Object[] fields) {
         try {
             Object ret = _class.newInstance();
@@ -222,6 +229,45 @@ public final class Reflection {
         }
     }
 
+    public static Object impose(Method _method, Object[] fields) {
+        try {
+            Class[] argTypes = _method.getParameterTypes();
+            Object[] args = new Object[argTypes.length];
+            int j = 0;
+            for(int i=0; i<args.length; i++) {
+                Object tgt = Reflection.lub(fields[i]);
+                if (argTypes[i] == String.class) tgt = Reflection.stringify(tgt);
+                // FUGLY
+                tgt = Reflection.coerce(tgt, argTypes[i]);
+                System.err.println("setting a " + argTypes[i].getName() + " to " + Reflection.show(tgt));
+                args[i] = tgt;
+            }
+            System.err.println("invoking " + _method + " with " + Reflection.show(args));
+            return _method.invoke(null, args);
+        } catch (Exception e) {
+            throw new RuntimeException(e);
+        }
+    }
+
+    public static Object impose(Constructor _ctor, Object[] fields) {
+        try {
+            Class[] argTypes = _ctor.getParameterTypes();
+            Object[] args = new Object[argTypes.length];
+            int j = 0;
+            for(int i=0; i<args.length; i++) {
+                Object tgt = Reflection.lub(fields[i]);
+                if (argTypes[i] == String.class) tgt = Reflection.stringify(tgt);
+                // FUGLY
+                tgt = Reflection.coerce(tgt, argTypes[i]);
+                System.err.println("setting a " + argTypes[i].getName() + " to " + Reflection.show(tgt));
+                args[i] = tgt;
+            }
+            return _ctor.newInstance(args);
+        } catch (Exception e) {
+            throw new RuntimeException(e);
+        }
+    }
+
     public static String stringify(Object o) {
         if (o==null) return "";
         if (!(o instanceof Object[])) return o.toString();
@@ -277,4 +323,65 @@ public final class Reflection {
         return o;
     }
 
+    public static abstract class Bindable {
+        public abstract String getSimpleName();
+        public abstract String toString();
+        public abstract <A extends Annotation> A getAnnotation(Class<A> c);
+        public abstract Object impose(Object[] fields);
+        public boolean isAnnotationPresent(Class<? extends Annotation> c) { return getAnnotation(c) != null; }
+
+        public abstract Annotation[][] getArgAnnotations();
+        public abstract String[]       getArgNames();
+
+        public static Bindable create(Object o) {
+            if (o instanceof Class) return new BindableClass((Class)o);
+            if (o instanceof Method) return new BindableMethod((Method)o);
+            if (o instanceof Constructor) return new BindableConstructor((Constructor)o);
+            return null;
+        }
+    }
+
+    private static class BindableMethod extends Bindable {
+        private final Method _method;
+        public String toString() { return "BindableMethod["+_method+"]"; }
+        public BindableMethod(Method _method) { this._method = _method; }
+        public String getSimpleName() { return _method.getName(); }
+        public <A extends Annotation> A getAnnotation(Class<A> c) { return _method.getAnnotation(c); }
+        public Object impose(Object[] fields) { return Reflection.impose(_method, fields); }
+        public Annotation[][] getArgAnnotations() { return _method.getParameterAnnotations(); }
+        public String[]       getArgNames() { return new String[_method.getParameterTypes().length]; }
+    }
+    private static class BindableConstructor extends Bindable {
+        private final Constructor _constructor;
+        public String toString() { return "BindableConstructor["+_constructor+"]"; }
+        public BindableConstructor(Constructor _constructor) { this._constructor = _constructor; }
+        public String getSimpleName() { return _constructor.getName(); }
+        public <A extends Annotation> A getAnnotation(Class<A> c) { return _constructor.getAnnotation(c); }
+        public Object impose(Object[] fields) { return Reflection.impose(_constructor, fields); }
+        public Annotation[][] getArgAnnotations() { return _constructor.getParameterAnnotations(); }
+        public String[]       getArgNames() { return new String[_constructor.getParameterTypes().length]; }
+    }
+    private static class BindableClass extends Bindable {
+        private final Class _class;
+        public String toString() { return "BindableClass["+_class+"]"; }
+        public BindableClass(Class _class) { this._class = _class; }
+        public String getSimpleName() { return _class.getSimpleName(); }
+        public <A extends Annotation> A getAnnotation(Class<A> c) { return (A)_class.getAnnotation(c); }
+        public Object impose(Object[] fields) { return Reflection.impose(_class, fields); }
+        public Annotation[][] getArgAnnotations() {
+            Field[] fields = _class.getFields();
+            Annotation[][] ret = new Annotation[fields.length][];
+            for(int i=0; i<fields.length; i++)
+                ret[i] = fields[i].getAnnotations();
+            return ret;
+        }
+        public String[]       getArgNames() {
+            Field[] fields = _class.getFields();
+            String[] ret = new String[fields.length];
+            for(int i=0; i<fields.length; i++)
+                ret[i] = fields[i].getName();
+            return ret;
+        }
+    }
+
 }