rt/vm/src/main/java/org/apidesign/vm4brwsr/ClassPath.java
author Jaroslav Tulach <jaroslav.tulach@apidesign.org>
Wed, 14 May 2014 16:09:03 +0200
branchclosure
changeset 1576 3e9bfed0d88e
parent 1575 08c8821c962d
child 1581 cf0f746f1a7f
permissions -rw-r--r--
loadBytes does not need the loader argument
jaroslav@644
     1
/**
jaroslav@644
     2
 * Back 2 Browser Bytecode Translator
jaroslav@644
     3
 * Copyright (C) 2012 Jaroslav Tulach <jaroslav.tulach@apidesign.org>
jaroslav@644
     4
 *
jaroslav@644
     5
 * This program is free software: you can redistribute it and/or modify
jaroslav@644
     6
 * it under the terms of the GNU General Public License as published by
jaroslav@644
     7
 * the Free Software Foundation, version 2 of the License.
jaroslav@644
     8
 *
jaroslav@644
     9
 * This program is distributed in the hope that it will be useful,
jaroslav@644
    10
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
jaroslav@644
    11
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
jaroslav@644
    12
 * GNU General Public License for more details.
jaroslav@644
    13
 *
jaroslav@644
    14
 * You should have received a copy of the GNU General Public License
jaroslav@644
    15
 * along with this program. Look for COPYING file in the top folder.
jaroslav@644
    16
 * If not, see http://opensource.org/licenses/GPL-2.0.
jaroslav@644
    17
 */
jaroslav@644
    18
package org.apidesign.vm4brwsr;
jaroslav@644
    19
jaroslav@672
    20
import java.io.ByteArrayInputStream;
jaroslav@644
    21
import java.io.IOException;
jaroslav@1551
    22
import java.io.InputStream;
jaroslav@644
    23
import java.net.URL;
jaroslav@1575
    24
import org.apidesign.bck2brwsr.core.Exported;
jaroslav@644
    25
import org.apidesign.bck2brwsr.core.JavaScriptBody;
jaroslav@644
    26
jaroslav@644
    27
/** Conversion from classpath to load function.
jaroslav@644
    28
 *
jaroslav@644
    29
 * @author Jaroslav Tulach <jtulach@netbeans.org>
jaroslav@644
    30
 */
jaroslav@1550
    31
final class ClassPath {
jaroslav@1550
    32
    private ClassPath() {
jaroslav@644
    33
    }
jaroslav@644
    34
    
jaroslav@644
    35
    public static void init() {
jaroslav@644
    36
    }
jaroslav@729
    37
    @JavaScriptBody(args = { "arr" }, body = "return arr.length;")
jaroslav@729
    38
    private static native int length(Object arr);
jaroslav@729
    39
    @JavaScriptBody(args = { "arr", "index" }, body = "return arr[index];")
jaroslav@729
    40
    private static native Object at(Object arr, int index);
jaroslav@729
    41
    @JavaScriptBody(args = { "arr", "index", "value" }, body = "arr[index] = value; return value;")
jaroslav@729
    42
    private static native Object set(Object arr, int index, Object value);
jaroslav@644
    43
    
jaroslav@1554
    44
    private static boolean doingToZip;
jaroslav@1575
    45
    
jaroslav@1575
    46
    
jaroslav@1576
    47
    @Exported static byte[] loadBytes(String resource, Object classpath, int skip) 
jaroslav@729
    48
    throws IOException, ClassNotFoundException {
jaroslav@729
    49
        for (int i = 0; i < length(classpath); i++) {
jaroslav@729
    50
            Object c = at(classpath, i);
jaroslav@1554
    51
            if (c instanceof String && !doingToZip) {
jaroslav@644
    52
                try {
jaroslav@1554
    53
                    doingToZip = true;
jaroslav@672
    54
                    String url = (String)c;
jaroslav@1551
    55
                    final Bck2Brwsr.Resources z = toZip(url);
jaroslav@729
    56
                    c = set(classpath, i, z);
jaroslav@1551
    57
                    final byte[] man = readBytes(z, "META-INF/MANIFEST.MF");
jaroslav@672
    58
                    if (man != null) {
jaroslav@702
    59
                        String mainClass = processClassPathAttr(man, url, classpath);
jaroslav@743
    60
//                        if (mainClass != null) {
jaroslav@743
    61
//                            Class.forName(mainClass);
jaroslav@743
    62
//                        }
jaroslav@672
    63
                    }
jaroslav@743
    64
                } catch (IOException ex) {
jaroslav@729
    65
                    set(classpath, i, ex);
jaroslav@746
    66
                    log("Cannot load " + c + " - " + ex.getClass().getName() + ":" + ex.getMessage());
jaroslav@1554
    67
                } finally {
jaroslav@1554
    68
                    doingToZip = false;
jaroslav@644
    69
                }
jaroslav@644
    70
            }
jaroslav@1576
    71
            if (resource != null) {
jaroslav@729
    72
                byte[] checkRes;
jaroslav@1551
    73
                if (c instanceof Bck2Brwsr.Resources) {
jaroslav@1576
    74
                    checkRes = readBytes((Bck2Brwsr.Resources)c, resource);
jaroslav@1387
    75
                    if (checkRes != null && --skip < 0) {
jaroslav@1387
    76
                        return checkRes;
jaroslav@1387
    77
                    }
jaroslav@729
    78
                } else {
jaroslav@1576
    79
                    checkRes = callFunction(c, resource, skip);
jaroslav@1387
    80
                    if (checkRes != null) {
jaroslav@1387
    81
                        return checkRes;
jaroslav@1387
    82
                    }
jaroslav@644
    83
                }
jaroslav@644
    84
            }
jaroslav@644
    85
        }
jaroslav@644
    86
        return null;
jaroslav@644
    87
    }
jaroslav@706
    88
    
jaroslav@1375
    89
    @JavaScriptBody(args = { "fn", "res", "skip" }, body = 
jaroslav@1375
    90
        "if (typeof fn === 'function') return fn(res, skip);\n"
jaroslav@729
    91
      + "return null;"
jaroslav@729
    92
    )
jaroslav@1375
    93
    private static native byte[] callFunction(Object fn, String res, int skip);
jaroslav@729
    94
    
jaroslav@1358
    95
    @JavaScriptBody(args = { "msg" }, body = "if (typeof console !== 'undefined') console.log(msg.toString());")
jaroslav@706
    96
    private static native void log(String msg);
jaroslav@706
    97
jaroslav@729
    98
    private static String processClassPathAttr(final byte[] man, String url, Object classpath) throws IOException {
jaroslav@702
    99
        try (ParseMan is = new ParseMan(new ByteArrayInputStream(man))) {
jaroslav@672
   100
            String cp = is.toString();
jaroslav@702
   101
            if (cp != null) {
jaroslav@702
   102
                cp = cp.trim();
jaroslav@702
   103
                for (int p = 0; p < cp.length();) {
jaroslav@702
   104
                    int n = cp.indexOf(' ', p);
jaroslav@702
   105
                    if (n == -1) {
jaroslav@702
   106
                        n = cp.length();
jaroslav@702
   107
                    }
jaroslav@702
   108
                    String el = cp.substring(p, n);
jaroslav@702
   109
                    URL u = new URL(new URL(url), el);
jaroslav@702
   110
                    classpath = addToArray(classpath, u.toString());
jaroslav@702
   111
                    p = n + 1;
jaroslav@702
   112
                }
jaroslav@672
   113
            }
jaroslav@702
   114
            return is.getMainClass();
jaroslav@672
   115
        }
jaroslav@672
   116
    }
jaroslav@672
   117
jaroslav@729
   118
    private static Object addToArray(Object arr, String value) {
jaroslav@729
   119
        final int last = length(arr);
jaroslav@729
   120
        Object ret = enlargeArray(arr, last + 1);
jaroslav@729
   121
        set(ret, last, value);
jaroslav@672
   122
        return ret;
jaroslav@672
   123
    }
jaroslav@672
   124
jaroslav@729
   125
    @JavaScriptBody(args = { "arr", "len" }, body = "while (arr.length < len) arr.push(null); return arr;")
jaroslav@729
   126
    private static native Object enlargeArray(Object arr, int len);
jaroslav@1554
   127
jaroslav@1551
   128
    private static Bck2Brwsr.Resources toZip(String path) throws IOException {
jaroslav@1551
   129
        URL u = new URL(path);
jaroslav@1551
   130
        byte[] zipData = (byte[]) u.getContent(new Class[]{byte[].class});
jaroslav@1551
   131
        Bck2Brwsr.Resources r;
jaroslav@1551
   132
        try {
jaroslav@1551
   133
            Class<?> fastJar = Class.forName("org.apidesign.bck2brwsr.vmzip.ZipResources");
jaroslav@1551
   134
            return (Bck2Brwsr.Resources) fastJar.getConstructor(byte[].class).newInstance(zipData);
jaroslav@1551
   135
        } catch (Exception ex) {
jaroslav@1551
   136
            log("Reading JARs is only possible with enum.zip module included: " + ex.getMessage());
jaroslav@1551
   137
            ex.printStackTrace();
jaroslav@1551
   138
            throw new IOException(ex);
jaroslav@1551
   139
        }
jaroslav@1551
   140
    }
jaroslav@1551
   141
    
jaroslav@1551
   142
    private static byte[] readBytes(Bck2Brwsr.Resources r, String res) throws IOException {
jaroslav@1551
   143
        InputStream is = r.get(res);
jaroslav@1551
   144
        if (is == null) {
jaroslav@1551
   145
            return null;
jaroslav@1551
   146
        }
jaroslav@1551
   147
        byte[] arr = new byte[is.available()];
jaroslav@1551
   148
        int off = 0;
jaroslav@1551
   149
        for (;;) {
jaroslav@1551
   150
            int len = is.read(arr, off, arr.length - off);
jaroslav@1551
   151
            if (len == -1) {
jaroslav@1551
   152
                break;
jaroslav@1551
   153
            }
jaroslav@1551
   154
            off += len;
jaroslav@1551
   155
        }
jaroslav@1551
   156
        is.close();
jaroslav@1551
   157
        return arr;
jaroslav@1551
   158
    }
jaroslav@644
   159
}