boot/src/main/java/org/apidesign/html/boot/impl/FnUtils.java
author Jaroslav Tulach <jaroslav.tulach@apidesign.org>
Wed, 26 Jun 2013 08:43:32 +0200
branchclassloader
changeset 163 2652760705d6
parent 161 ea5ca9cc685d
child 171 54ac82353158
permissions -rw-r--r--
Loading of external JavaScript files simplified
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@163
    23
import java.io.InputStream;
jaroslav@163
    24
import java.io.InputStreamReader;
jaroslav@163
    25
import java.io.Reader;
jaroslav@160
    26
import java.lang.reflect.Method;
jaroslav@123
    27
import java.net.URL;
jaroslav@123
    28
import java.util.ArrayList;
jaroslav@123
    29
import java.util.Collections;
jaroslav@123
    30
import java.util.Enumeration;
jaroslav@123
    31
import java.util.List;
jaroslav@123
    32
import org.apidesign.html.boot.spi.Fn;
jaroslav@161
    33
import org.objectweb.asm.Type;
jaroslav@123
    34
jaroslav@123
    35
/**
jaroslav@123
    36
 *
jaroslav@123
    37
 * @author Jaroslav Tulach <jtulach@netbeans.org>
jaroslav@123
    38
 */
jaroslav@123
    39
public final class FnUtils {
jaroslav@123
    40
    private FnUtils() {
jaroslav@123
    41
    }
jaroslav@123
    42
jaroslav@123
    43
    public static Fn define(Class<?> caller, String code, String... names) {
jaroslav@123
    44
        JsClassLoader cl = (JsClassLoader)caller.getClassLoader();
jaroslav@123
    45
        return cl.defineFn(code, names);
jaroslav@123
    46
    }
jaroslav@123
    47
jaroslav@128
    48
    public static ClassLoader newLoader(final FindResources f, final Fn.Presenter d, ClassLoader parent) {
jaroslav@123
    49
        return new JsClassLoader(parent) {
jaroslav@123
    50
            @Override
jaroslav@123
    51
            protected URL findResource(String name) {
jaroslav@123
    52
                List<URL> l = res(name, true);
jaroslav@123
    53
                return l.isEmpty() ? null : l.get(0);
jaroslav@123
    54
            }
jaroslav@123
    55
            
jaroslav@123
    56
            @Override
jaroslav@123
    57
            protected Enumeration<URL> findResources(String name) {
jaroslav@123
    58
                return Collections.enumeration(res(name, false));
jaroslav@123
    59
            }
jaroslav@123
    60
            
jaroslav@123
    61
            private List<URL> res(String name, boolean oneIsEnough) {
jaroslav@123
    62
                List<URL> l = new ArrayList<URL>();
jaroslav@123
    63
                f.findResources(name, l, oneIsEnough);
jaroslav@123
    64
                return l;
jaroslav@123
    65
            }
jaroslav@123
    66
            
jaroslav@123
    67
            @Override
jaroslav@123
    68
            protected Fn defineFn(String code, String... names) {
jaroslav@123
    69
                return d.defineFn(code, names);
jaroslav@123
    70
            }
jaroslav@163
    71
jaroslav@163
    72
            @Override
jaroslav@163
    73
            protected void loadScript(Reader code) throws Exception {
jaroslav@163
    74
                d.loadScript(code);
jaroslav@163
    75
            }
jaroslav@123
    76
        };
jaroslav@123
    77
    }
jaroslav@160
    78
jaroslav@160
    79
    static String callback(String body, ClassLoader loader) {
jaroslav@160
    80
        try {
jaroslav@160
    81
            return callbackImpl(body, loader);
jaroslav@160
    82
        } catch (ClassNotFoundException ex) {
jaroslav@160
    83
            throw new IllegalStateException("Can't parse " + body, ex);
jaroslav@160
    84
        } catch (NoSuchMethodException ex) {
jaroslav@160
    85
            throw new IllegalStateException("Can't parse " + body, ex);
jaroslav@160
    86
        }
jaroslav@160
    87
    }
jaroslav@160
    88
    
jaroslav@160
    89
    private static String callbackImpl(String body, ClassLoader loader)
jaroslav@160
    90
    throws ClassNotFoundException, NoSuchMethodException {
jaroslav@160
    91
        StringBuilder sb = new StringBuilder();
jaroslav@160
    92
        int pos = 0;
jaroslav@160
    93
        for (;;) {
jaroslav@160
    94
            int next = body.indexOf(".@", pos);
jaroslav@160
    95
            if (next == -1) {
jaroslav@160
    96
                sb.append(body.substring(pos));
jaroslav@160
    97
                return sb.toString();
jaroslav@160
    98
            }
jaroslav@160
    99
            sb.append(body.substring(pos, next));
jaroslav@160
   100
            
jaroslav@160
   101
            int sigBeg = body.indexOf('(', next);
jaroslav@160
   102
            int sigEnd = body.indexOf(')', sigBeg);
jaroslav@160
   103
            
jaroslav@160
   104
            int colon4 = body.indexOf("::", next);
jaroslav@160
   105
            
jaroslav@160
   106
            if (sigBeg == -1 || sigEnd == -1 || colon4 == -1) {
jaroslav@160
   107
                throw new IllegalStateException("Malformed body " + body);
jaroslav@160
   108
            }
jaroslav@160
   109
            
jaroslav@160
   110
            String fqn = body.substring(next + 2, colon4);
jaroslav@160
   111
            String method = body.substring(colon4 + 2, sigBeg);
jaroslav@161
   112
            String params = body.substring(sigBeg, sigEnd + 1);
jaroslav@160
   113
            
jaroslav@160
   114
            Class<?> clazz = Class.forName(fqn, false, loader);
jaroslav@161
   115
            final Type[] argTps = Type.getArgumentTypes(params);
jaroslav@161
   116
            Class<?>[] argCls = new Class<?>[argTps.length];
jaroslav@161
   117
            for (int i = 0; i < argCls.length; i++) {
jaroslav@161
   118
                argCls[i] = toClass(argTps[i], loader);
jaroslav@161
   119
            }
jaroslav@161
   120
            Method m = clazz.getMethod(method, argCls);
jaroslav@160
   121
            
jaroslav@160
   122
            sb.append("['").append(m.getName()).append("(");
jaroslav@160
   123
            String sep = "";
jaroslav@160
   124
            for (Class<?> pt : m.getParameterTypes()) {
jaroslav@160
   125
                sb.append(sep).append(pt.getName());
jaroslav@160
   126
                sep = ",";
jaroslav@160
   127
            }
jaroslav@160
   128
            sb.append(")']");
jaroslav@160
   129
            
jaroslav@160
   130
            pos = sigEnd + 1;
jaroslav@160
   131
        }
jaroslav@160
   132
    }
jaroslav@160
   133
jaroslav@161
   134
    private static Class<?> toClass(final Type t, ClassLoader loader) throws ClassNotFoundException {
jaroslav@161
   135
        if (t == Type.INT_TYPE) {
jaroslav@161
   136
            return Integer.TYPE;
jaroslav@161
   137
        } else if (t == Type.VOID_TYPE) {
jaroslav@161
   138
            return Void.TYPE;
jaroslav@161
   139
        } else if (t == Type.BOOLEAN_TYPE) {
jaroslav@161
   140
            return Boolean.TYPE;
jaroslav@161
   141
        } else if (t == Type.BYTE_TYPE) {
jaroslav@161
   142
            return Byte.TYPE;
jaroslav@161
   143
        } else if (t == Type.CHAR_TYPE) {
jaroslav@161
   144
            return Character.TYPE;
jaroslav@161
   145
        } else if (t == Type.SHORT_TYPE) {
jaroslav@161
   146
            return Short.TYPE;
jaroslav@161
   147
        } else if (t == Type.DOUBLE_TYPE) {
jaroslav@161
   148
            return Double.TYPE;
jaroslav@161
   149
        } else if (t == Type.FLOAT_TYPE) {
jaroslav@161
   150
            return Float.TYPE;
jaroslav@161
   151
        } else if (t == Type.LONG_TYPE) {
jaroslav@161
   152
            return Long.TYPE;
jaroslav@161
   153
        }
jaroslav@161
   154
        return Class.forName(t.getClassName(), false, loader);
jaroslav@161
   155
    }
jaroslav@163
   156
jaroslav@163
   157
    static void loadScript(JsClassLoader jcl, String resource) {
jaroslav@163
   158
        final InputStream script = jcl.getResourceAsStream(resource);
jaroslav@163
   159
        if (script == null) {
jaroslav@163
   160
            throw new NullPointerException("Can't find " + resource);
jaroslav@163
   161
        }
jaroslav@163
   162
        try {
jaroslav@163
   163
            Reader isr = null;
jaroslav@163
   164
            try {
jaroslav@163
   165
                isr = new InputStreamReader(script, "UTF-8");
jaroslav@163
   166
                jcl.loadScript(isr);
jaroslav@163
   167
            } finally {
jaroslav@163
   168
                if (isr != null) {
jaroslav@163
   169
                    isr.close();
jaroslav@163
   170
                }
jaroslav@163
   171
            }
jaroslav@163
   172
        } catch (Exception ex) {
jaroslav@163
   173
            throw new IllegalStateException("Can't execute " + resource, ex);
jaroslav@163
   174
        } 
jaroslav@163
   175
    }
jaroslav@123
   176
}