rt/launcher/src/main/java/org/apidesign/bck2brwsr/launcher/JSLauncher.java
author Jaroslav Tulach <jaroslav.tulach@apidesign.org>
Tue, 26 Feb 2013 16:54:16 +0100
changeset 772 d382dacfd73f
parent 622 launcher/src/main/java/org/apidesign/bck2brwsr/launcher/JSLauncher.java@a07253cf2ca4
child 1020 a6bacea2518f
permissions -rw-r--r--
Moving modules around so the runtime is under one master pom and can be built without building other modules that are in the repository
     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.bck2brwsr.launcher;
    19 
    20 import org.apidesign.bck2brwsr.launcher.impl.Console;
    21 import java.io.IOException;
    22 import java.io.InputStream;
    23 import java.net.URL;
    24 import java.util.Enumeration;
    25 import java.util.LinkedHashSet;
    26 import java.util.Set;
    27 import java.util.logging.Level;
    28 import java.util.logging.Logger;
    29 import javax.script.Invocable;
    30 import javax.script.ScriptEngine;
    31 import javax.script.ScriptEngineManager;
    32 import javax.script.ScriptException;
    33 import org.apidesign.vm4brwsr.Bck2Brwsr;
    34 
    35 /**
    36  * Tests execution in Java's internal scripting engine.
    37  */
    38 final class JSLauncher extends Launcher {
    39     private static final Logger LOG = Logger.getLogger(JSLauncher.class.getName());
    40     private Set<ClassLoader> loaders = new LinkedHashSet<>();
    41     private final Res resources = new Res();
    42     private Invocable code;
    43     private StringBuilder codeSeq;
    44     private Object console;
    45     
    46     
    47     @Override InvocationContext runMethod(InvocationContext mi) {
    48         loaders.add(mi.clazz.getClassLoader());
    49         try {
    50             long time = System.currentTimeMillis();
    51             LOG.log(Level.FINE, "Invoking {0}.{1}", new Object[]{mi.clazz.getName(), mi.methodName});
    52             String res = code.invokeMethod(
    53                 console,
    54                 "invoke__Ljava_lang_String_2Ljava_lang_String_2Ljava_lang_String_2",
    55                 mi.clazz.getName(), mi.methodName).toString();
    56             time = System.currentTimeMillis() - time;
    57             LOG.log(Level.FINE, "Resut of {0}.{1} = {2} in {3} ms", new Object[]{mi.clazz.getName(), mi.methodName, res, time});
    58             mi.result(res, null);
    59         } catch (ScriptException | NoSuchMethodException ex) {
    60             mi.result(null, ex);
    61         }
    62         return mi;
    63     }
    64     
    65     public void addClassLoader(ClassLoader url) {
    66         this.loaders.add(url);
    67     }
    68 
    69     @Override
    70     public void initialize() throws IOException {
    71         try {
    72             initRhino();
    73         } catch (Exception ex) {
    74             if (ex instanceof IOException) {
    75                 throw (IOException)ex;
    76             }
    77             if (ex instanceof RuntimeException) {
    78                 throw (RuntimeException)ex;
    79             }
    80             throw new IOException(ex);
    81         }
    82     }
    83     
    84     private void initRhino() throws IOException, ScriptException, NoSuchMethodException {
    85         StringBuilder sb = new StringBuilder();
    86         Bck2Brwsr.generate(sb, new Res());
    87 
    88         ScriptEngineManager sem = new ScriptEngineManager();
    89         ScriptEngine mach = sem.getEngineByExtension("js");
    90 
    91         sb.append(
    92               "\nvar vm = new bck2brwsr(org.apidesign.bck2brwsr.launcher.impl.Console.read);"
    93             + "\nfunction initVM() { return vm; };"
    94             + "\n");
    95 
    96         Object res = mach.eval(sb.toString());
    97         if (!(mach instanceof Invocable)) {
    98             throw new IOException("It is invocable object: " + res);
    99         }
   100         code = (Invocable) mach;
   101         codeSeq = sb;
   102         
   103         Object vm = code.invokeFunction("initVM");
   104         console = code.invokeMethod(vm, "loadClass", Console.class.getName());
   105     }
   106 
   107     @Override
   108     public void shutdown() throws IOException {
   109     }
   110 
   111     @Override
   112     public String toString() {
   113         return codeSeq.toString();
   114     }
   115     
   116     private class Res implements Bck2Brwsr.Resources {
   117         @Override
   118         public InputStream get(String resource) throws IOException {
   119             for (ClassLoader l : loaders) {
   120                 URL u = null;
   121                 Enumeration<URL> en = l.getResources(resource);
   122                 while (en.hasMoreElements()) {
   123                     u = en.nextElement();
   124                 }
   125                 if (u != null) {
   126                     return u.openStream();
   127                 }
   128             }
   129             throw new IOException("Can't find " + resource);
   130         }
   131     }
   132 }