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