boot/src/main/java/org/apidesign/html/boot/impl/FnUtils.java
author Jaroslav Tulach <jaroslav.tulach@apidesign.org>
Tue, 25 Jun 2013 21:09:48 +0200
branchclassloader
changeset 161 ea5ca9cc685d
parent 160 b5a06e2b3b92
child 163 2652760705d6
permissions -rw-r--r--
Can call method with two integer arguments from JavaScript
jaroslav@123
     1
/**
jaroslav@123
     2
 * HTML via Java(tm) Language Bindings
jaroslav@123
     3
 * Copyright (C) 2013 Jaroslav Tulach <jaroslav.tulach@apidesign.org>
jaroslav@123
     4
 *
jaroslav@123
     5
 * This program is free software: you can redistribute it and/or modify
jaroslav@123
     6
 * it under the terms of the GNU General Public License as published by
jaroslav@123
     7
 * the Free Software Foundation, version 2 of the License.
jaroslav@123
     8
 *
jaroslav@123
     9
 * This program is distributed in the hope that it will be useful,
jaroslav@123
    10
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
jaroslav@123
    11
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
jaroslav@123
    12
 * GNU General Public License for more details. apidesign.org
jaroslav@123
    13
 * designates this particular file as subject to the
jaroslav@123
    14
 * "Classpath" exception as provided by apidesign.org
jaroslav@123
    15
 * in the License file that accompanied this code.
jaroslav@123
    16
 *
jaroslav@123
    17
 * You should have received a copy of the GNU General Public License
jaroslav@123
    18
 * along with this program. Look for COPYING file in the top folder.
jaroslav@123
    19
 * If not, see http://wiki.apidesign.org/wiki/GPLwithClassPathException
jaroslav@123
    20
 */
jaroslav@123
    21
package org.apidesign.html.boot.impl;
jaroslav@123
    22
jaroslav@160
    23
import java.lang.reflect.Method;
jaroslav@123
    24
import java.net.URL;
jaroslav@123
    25
import java.util.ArrayList;
jaroslav@123
    26
import java.util.Collections;
jaroslav@123
    27
import java.util.Enumeration;
jaroslav@123
    28
import java.util.List;
jaroslav@123
    29
import org.apidesign.html.boot.spi.Fn;
jaroslav@161
    30
import org.objectweb.asm.Type;
jaroslav@123
    31
jaroslav@123
    32
/**
jaroslav@123
    33
 *
jaroslav@123
    34
 * @author Jaroslav Tulach <jtulach@netbeans.org>
jaroslav@123
    35
 */
jaroslav@123
    36
public final class FnUtils {
jaroslav@123
    37
    private FnUtils() {
jaroslav@123
    38
    }
jaroslav@123
    39
jaroslav@123
    40
    public static Fn define(Class<?> caller, String code, String... names) {
jaroslav@123
    41
        JsClassLoader cl = (JsClassLoader)caller.getClassLoader();
jaroslav@123
    42
        return cl.defineFn(code, names);
jaroslav@123
    43
    }
jaroslav@123
    44
jaroslav@128
    45
    public static ClassLoader newLoader(final FindResources f, final Fn.Presenter d, ClassLoader parent) {
jaroslav@123
    46
        return new JsClassLoader(parent) {
jaroslav@123
    47
            @Override
jaroslav@123
    48
            protected URL findResource(String name) {
jaroslav@123
    49
                List<URL> l = res(name, true);
jaroslav@123
    50
                return l.isEmpty() ? null : l.get(0);
jaroslav@123
    51
            }
jaroslav@123
    52
            
jaroslav@123
    53
            @Override
jaroslav@123
    54
            protected Enumeration<URL> findResources(String name) {
jaroslav@123
    55
                return Collections.enumeration(res(name, false));
jaroslav@123
    56
            }
jaroslav@123
    57
            
jaroslav@123
    58
            private List<URL> res(String name, boolean oneIsEnough) {
jaroslav@123
    59
                List<URL> l = new ArrayList<URL>();
jaroslav@123
    60
                f.findResources(name, l, oneIsEnough);
jaroslav@123
    61
                return l;
jaroslav@123
    62
            }
jaroslav@123
    63
            
jaroslav@123
    64
            @Override
jaroslav@123
    65
            protected Fn defineFn(String code, String... names) {
jaroslav@123
    66
                return d.defineFn(code, names);
jaroslav@123
    67
            }
jaroslav@123
    68
        };
jaroslav@123
    69
    }
jaroslav@160
    70
jaroslav@160
    71
    static String callback(String body, ClassLoader loader) {
jaroslav@160
    72
        try {
jaroslav@160
    73
            return callbackImpl(body, loader);
jaroslav@160
    74
        } catch (ClassNotFoundException ex) {
jaroslav@160
    75
            throw new IllegalStateException("Can't parse " + body, ex);
jaroslav@160
    76
        } catch (NoSuchMethodException ex) {
jaroslav@160
    77
            throw new IllegalStateException("Can't parse " + body, ex);
jaroslav@160
    78
        }
jaroslav@160
    79
    }
jaroslav@160
    80
    
jaroslav@160
    81
    private static String callbackImpl(String body, ClassLoader loader)
jaroslav@160
    82
    throws ClassNotFoundException, NoSuchMethodException {
jaroslav@160
    83
        StringBuilder sb = new StringBuilder();
jaroslav@160
    84
        int pos = 0;
jaroslav@160
    85
        for (;;) {
jaroslav@160
    86
            int next = body.indexOf(".@", pos);
jaroslav@160
    87
            if (next == -1) {
jaroslav@160
    88
                sb.append(body.substring(pos));
jaroslav@160
    89
                return sb.toString();
jaroslav@160
    90
            }
jaroslav@160
    91
            sb.append(body.substring(pos, next));
jaroslav@160
    92
            
jaroslav@160
    93
            int sigBeg = body.indexOf('(', next);
jaroslav@160
    94
            int sigEnd = body.indexOf(')', sigBeg);
jaroslav@160
    95
            
jaroslav@160
    96
            int colon4 = body.indexOf("::", next);
jaroslav@160
    97
            
jaroslav@160
    98
            if (sigBeg == -1 || sigEnd == -1 || colon4 == -1) {
jaroslav@160
    99
                throw new IllegalStateException("Malformed body " + body);
jaroslav@160
   100
            }
jaroslav@160
   101
            
jaroslav@160
   102
            String fqn = body.substring(next + 2, colon4);
jaroslav@160
   103
            String method = body.substring(colon4 + 2, sigBeg);
jaroslav@161
   104
            String params = body.substring(sigBeg, sigEnd + 1);
jaroslav@160
   105
            
jaroslav@160
   106
            Class<?> clazz = Class.forName(fqn, false, loader);
jaroslav@161
   107
            final Type[] argTps = Type.getArgumentTypes(params);
jaroslav@161
   108
            Class<?>[] argCls = new Class<?>[argTps.length];
jaroslav@161
   109
            for (int i = 0; i < argCls.length; i++) {
jaroslav@161
   110
                argCls[i] = toClass(argTps[i], loader);
jaroslav@161
   111
            }
jaroslav@161
   112
            Method m = clazz.getMethod(method, argCls);
jaroslav@160
   113
            
jaroslav@160
   114
            sb.append("['").append(m.getName()).append("(");
jaroslav@160
   115
            String sep = "";
jaroslav@160
   116
            for (Class<?> pt : m.getParameterTypes()) {
jaroslav@160
   117
                sb.append(sep).append(pt.getName());
jaroslav@160
   118
                sep = ",";
jaroslav@160
   119
            }
jaroslav@160
   120
            sb.append(")']");
jaroslav@160
   121
            
jaroslav@160
   122
            pos = sigEnd + 1;
jaroslav@160
   123
        }
jaroslav@160
   124
    }
jaroslav@160
   125
jaroslav@161
   126
    private static Class<?> toClass(final Type t, ClassLoader loader) throws ClassNotFoundException {
jaroslav@161
   127
        if (t == Type.INT_TYPE) {
jaroslav@161
   128
            return Integer.TYPE;
jaroslav@161
   129
        } else if (t == Type.VOID_TYPE) {
jaroslav@161
   130
            return Void.TYPE;
jaroslav@161
   131
        } else if (t == Type.BOOLEAN_TYPE) {
jaroslav@161
   132
            return Boolean.TYPE;
jaroslav@161
   133
        } else if (t == Type.BYTE_TYPE) {
jaroslav@161
   134
            return Byte.TYPE;
jaroslav@161
   135
        } else if (t == Type.CHAR_TYPE) {
jaroslav@161
   136
            return Character.TYPE;
jaroslav@161
   137
        } else if (t == Type.SHORT_TYPE) {
jaroslav@161
   138
            return Short.TYPE;
jaroslav@161
   139
        } else if (t == Type.DOUBLE_TYPE) {
jaroslav@161
   140
            return Double.TYPE;
jaroslav@161
   141
        } else if (t == Type.FLOAT_TYPE) {
jaroslav@161
   142
            return Float.TYPE;
jaroslav@161
   143
        } else if (t == Type.LONG_TYPE) {
jaroslav@161
   144
            return Long.TYPE;
jaroslav@161
   145
        }
jaroslav@161
   146
        return Class.forName(t.getClassName(), false, loader);
jaroslav@161
   147
    }
jaroslav@123
   148
    
jaroslav@123
   149
}