launcher/http/src/main/java/org/apidesign/bck2brwsr/launcher/CompileCP.java
author Jaroslav Tulach <jaroslav.tulach@apidesign.org>
Wed, 21 May 2014 23:42:24 +0200
branchclosure
changeset 1583 89b6b369c13d
parent 1552 5a840853d1a8
child 1586 d4ee65642d8d
permissions -rw-r--r--
Support for exporting whole packages. addRootClasses defined in terms of addClasses and addExported.
jaroslav@1489
     1
/**
jaroslav@1489
     2
 * Back 2 Browser Bytecode Translator
jaroslav@1489
     3
 * Copyright (C) 2012 Jaroslav Tulach <jaroslav.tulach@apidesign.org>
jaroslav@1489
     4
 *
jaroslav@1489
     5
 * This program is free software: you can redistribute it and/or modify
jaroslav@1489
     6
 * it under the terms of the GNU General Public License as published by
jaroslav@1489
     7
 * the Free Software Foundation, version 2 of the License.
jaroslav@1489
     8
 *
jaroslav@1489
     9
 * This program is distributed in the hope that it will be useful,
jaroslav@1489
    10
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
jaroslav@1489
    11
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
jaroslav@1489
    12
 * GNU General Public License for more details.
jaroslav@1489
    13
 *
jaroslav@1489
    14
 * You should have received a copy of the GNU General Public License
jaroslav@1489
    15
 * along with this program. Look for COPYING file in the top folder.
jaroslav@1489
    16
 * If not, see http://opensource.org/licenses/GPL-2.0.
jaroslav@1489
    17
 */
jaroslav@1489
    18
package org.apidesign.bck2brwsr.launcher;
jaroslav@1489
    19
jaroslav@1525
    20
import java.io.BufferedReader;
jaroslav@1489
    21
import java.io.File;
jaroslav@1489
    22
import java.io.IOException;
jaroslav@1489
    23
import java.io.InputStream;
jaroslav@1525
    24
import java.io.InputStreamReader;
jaroslav@1489
    25
import java.io.StringWriter;
jaroslav@1492
    26
import java.net.JarURLConnection;
jaroslav@1489
    27
import java.net.URISyntaxException;
jaroslav@1489
    28
import java.net.URL;
jaroslav@1489
    29
import java.util.ArrayList;
jaroslav@1489
    30
import java.util.Enumeration;
jaroslav@1525
    31
import java.util.HashSet;
jaroslav@1489
    32
import java.util.List;
jaroslav@1525
    33
import java.util.Set;
jaroslav@1489
    34
import java.util.jar.JarEntry;
jaroslav@1489
    35
import java.util.jar.JarFile;
jaroslav@1505
    36
import java.util.logging.Level;
jaroslav@1505
    37
import java.util.logging.Logger;
jaroslav@1489
    38
import java.util.zip.ZipEntry;
jaroslav@1492
    39
import org.apidesign.bck2brwsr.launcher.BaseHTTPLauncher.Res;
jaroslav@1489
    40
import org.apidesign.vm4brwsr.Bck2Brwsr;
jaroslav@1489
    41
jaroslav@1489
    42
/**
jaroslav@1489
    43
 *
jaroslav@1489
    44
 * @author Jaroslav Tulach
jaroslav@1489
    45
 */
jaroslav@1489
    46
class CompileCP {
jaroslav@1523
    47
    private static final Logger LOG = Logger.getLogger(CompileCP.class.getName());
jaroslav@1525
    48
    static String compileJAR(final JarFile jar, Set<String> testClasses) 
jaroslav@1525
    49
    throws IOException {
jaroslav@1489
    50
        List<String> arr = new ArrayList<>();
jaroslav@1489
    51
        List<String> classes = new ArrayList<>();
jaroslav@1525
    52
        Set<String> keep = new HashSet<String>(testClasses);
jaroslav@1583
    53
        listJAR(jar, classes, arr, keep);
jaroslav@1525
    54
        
jaroslav@1489
    55
        StringWriter w = new StringWriter();
jaroslav@1489
    56
        try {
jaroslav@1489
    57
            class JarRes extends EmulationResources implements Bck2Brwsr.Resources {
jaroslav@1489
    58
                @Override
jaroslav@1489
    59
                public InputStream get(String resource) throws IOException {
jaroslav@1489
    60
                    InputStream is = jar.getInputStream(new ZipEntry(resource));
jaroslav@1489
    61
                    return is == null ? super.get(resource) : is;
jaroslav@1489
    62
                }
jaroslav@1489
    63
            }
jaroslav@1489
    64
            
jaroslav@1489
    65
            Bck2Brwsr.newCompiler()
jaroslav@1491
    66
                .addClasses(classes.toArray(new String[0]))
jaroslav@1583
    67
                .addExported(keep.toArray(new String[0]))
jaroslav@1498
    68
                .addResources(arr.toArray(new String[0]))
jaroslav@1491
    69
                .library(true)
jaroslav@1489
    70
                .resources(new JarRes())
jaroslav@1489
    71
                .generate(w);
jaroslav@1489
    72
            w.flush();
jaroslav@1489
    73
            return w.toString();
jaroslav@1523
    74
        } catch (IOException ex) {
jaroslav@1523
    75
            throw ex;
jaroslav@1489
    76
        } catch (Throwable ex) {
jaroslav@1489
    77
            throw new IOException("Cannot compile: ", ex);
jaroslav@1489
    78
        } finally {
jaroslav@1489
    79
            w.close();
jaroslav@1489
    80
        }
jaroslav@1489
    81
    }
jaroslav@1489
    82
    
jaroslav@1505
    83
    static String compileFromClassPath(URL u, final Res r) throws IOException {
jaroslav@1505
    84
        File f;
jaroslav@1505
    85
        try {
jaroslav@1505
    86
            f = new File(u.toURI());
jaroslav@1505
    87
        } catch (URISyntaxException ex) {
jaroslav@1505
    88
            throw new IOException(ex);
jaroslav@1505
    89
        }
jaroslav@1489
    90
        for (String s : System.getProperty("java.class.path").split(File.pathSeparator)) {
jaroslav@1489
    91
            if (!f.getPath().startsWith(s)) {
jaroslav@1489
    92
                continue;
jaroslav@1489
    93
            }
jaroslav@1489
    94
            File root = new File(s);
jaroslav@1489
    95
            List<String> arr = new ArrayList<>();
jaroslav@1489
    96
            List<String> classes = new ArrayList<>();
jaroslav@1489
    97
            listDir(root, null, classes, arr);
jaroslav@1489
    98
            StringWriter w = new StringWriter();
jaroslav@1489
    99
            try {
jaroslav@1489
   100
                Bck2Brwsr.newCompiler()
jaroslav@1489
   101
                    .addRootClasses(classes.toArray(new String[0]))
jaroslav@1498
   102
                    .addResources(arr.toArray(new String[0]))
jaroslav@1491
   103
                    .library(true)
jaroslav@1504
   104
                    .resources(new EmulationResources() {
jaroslav@1504
   105
                        @Override
jaroslav@1504
   106
                        public InputStream get(String resource) throws IOException {
jaroslav@1517
   107
                            if (r != null) {
jaroslav@1517
   108
                                final URL url = r.get(resource, 0);
jaroslav@1517
   109
                                return url == null ? null : url.openStream();
jaroslav@1517
   110
                            }
jaroslav@1517
   111
                            return super.get(resource);
jaroslav@1504
   112
                        }
jaroslav@1504
   113
                    })
jaroslav@1489
   114
                    .generate(w);
jaroslav@1489
   115
                w.flush();
jaroslav@1489
   116
                return w.toString();
jaroslav@1489
   117
            } catch (ClassFormatError ex) {
jaroslav@1489
   118
                throw new IOException(ex);
jaroslav@1489
   119
            } finally {
jaroslav@1489
   120
                w.close();
jaroslav@1489
   121
            }
jaroslav@1489
   122
        }
jaroslav@1489
   123
        return null;
jaroslav@1489
   124
    }
jaroslav@1489
   125
    
jaroslav@1525
   126
    private static void listJAR(
jaroslav@1525
   127
        JarFile j, List<String> classes,
jaroslav@1583
   128
        List<String> resources, Set<String> keep
jaroslav@1525
   129
    ) throws IOException {
jaroslav@1489
   130
        Enumeration<JarEntry> en = j.entries();
jaroslav@1489
   131
        while (en.hasMoreElements()) {
jaroslav@1489
   132
            JarEntry e = en.nextElement();
jaroslav@1489
   133
            final String n = e.getName();
jaroslav@1489
   134
            if (n.endsWith("/")) {
jaroslav@1489
   135
                continue;
jaroslav@1489
   136
            }
jaroslav@1489
   137
            int last = n.lastIndexOf('/');
jaroslav@1489
   138
            String pkg = n.substring(0, last + 1);
jaroslav@1489
   139
            if (skipPkg(pkg)) {
jaroslav@1489
   140
                continue;
jaroslav@1489
   141
            }
jaroslav@1489
   142
            if (n.endsWith(".class")) {
jaroslav@1489
   143
                classes.add(n.substring(0, n.length() - 6));
jaroslav@1489
   144
            } else {
jaroslav@1489
   145
                resources.add(n);
jaroslav@1525
   146
                if (n.startsWith("META-INF/services/") && keep != null) {
jaroslav@1525
   147
                    BufferedReader r = new BufferedReader(new InputStreamReader(j.getInputStream(e)));
jaroslav@1525
   148
                    for (;;) {
jaroslav@1525
   149
                        String l = r.readLine();
jaroslav@1525
   150
                        if (l == null) {
jaroslav@1525
   151
                            break;
jaroslav@1525
   152
                        }
jaroslav@1525
   153
                        if (l.startsWith("#")) {
jaroslav@1525
   154
                            continue;
jaroslav@1525
   155
                        }
jaroslav@1525
   156
                        keep.add(l.replace('.', '/'));
jaroslav@1525
   157
                    }
jaroslav@1525
   158
                }
jaroslav@1525
   159
            }
jaroslav@1525
   160
        }
jaroslav@1525
   161
        String exp = j.getManifest().getMainAttributes().getValue("Export-Package");
jaroslav@1583
   162
        if (exp != null && keep != null) {
jaroslav@1525
   163
            for (String def : exp.split(",")) {
jaroslav@1525
   164
                for (String sep : def.split(";")) {
jaroslav@1583
   165
                    keep.add(sep.replace('.', '/') + "/");
jaroslav@1525
   166
                    break;
jaroslav@1525
   167
                }
jaroslav@1489
   168
            }
jaroslav@1489
   169
        }
jaroslav@1489
   170
    }
jaroslav@1489
   171
jaroslav@1489
   172
    private static boolean skipPkg(String pkg) {
jaroslav@1489
   173
        return pkg.equals("org/apidesign/bck2brwsr/launcher/");
jaroslav@1489
   174
    }
jaroslav@1489
   175
    
jaroslav@1489
   176
    private static void listDir(File f, String pref, List<String> classes, List<String> resources) throws IOException {
jaroslav@1489
   177
        File[] arr = f.listFiles();
jaroslav@1489
   178
        if (arr == null) {
jaroslav@1489
   179
            if (f.getName().endsWith(".class")) {
jaroslav@1489
   180
                classes.add(pref + f.getName().substring(0, f.getName().length() - 6));
jaroslav@1489
   181
            } else {
jaroslav@1489
   182
                resources.add(pref + f.getName());
jaroslav@1489
   183
            }
jaroslav@1489
   184
        } else {
jaroslav@1489
   185
            for (File ch : arr) {
jaroslav@1489
   186
                
jaroslav@1489
   187
                listDir(ch, pref == null ? "" : pref + f.getName() + "/", classes, resources);
jaroslav@1489
   188
            }
jaroslav@1489
   189
        }
jaroslav@1489
   190
    }
jaroslav@1489
   191
jaroslav@1504
   192
    static void compileVM(StringBuilder sb, final Res r) throws IOException {
jaroslav@1492
   193
        List<String> arr = new ArrayList<>();
jaroslav@1492
   194
        List<String> classes = new ArrayList<>();
jaroslav@1552
   195
jaroslav@1552
   196
        {
jaroslav@1552
   197
            URL u = r.get(InterruptedException.class.getName().replace('.', '/') + ".class", 0);
jaroslav@1552
   198
            JarURLConnection juc = (JarURLConnection)u.openConnection();
jaroslav@1583
   199
            listJAR(juc.getJarFile(), classes, arr, null);
jaroslav@1552
   200
        }
jaroslav@1552
   201
        {
jaroslav@1552
   202
            URL u = r.get(Bck2Brwsr.class.getName().replace('.', '/') + ".class", 0);
jaroslav@1552
   203
            JarURLConnection juc = (JarURLConnection)u.openConnection();
jaroslav@1583
   204
            listJAR(juc.getJarFile(), classes, arr, null);
jaroslav@1552
   205
        }
jaroslav@1492
   206
jaroslav@1492
   207
        Bck2Brwsr.newCompiler().addRootClasses(classes.toArray(new String[0]))
jaroslav@1504
   208
            .resources(new Bck2Brwsr.Resources() {
jaroslav@1504
   209
                @Override
jaroslav@1504
   210
                public InputStream get(String resource) throws IOException {
jaroslav@1523
   211
                    final URL url = r.get(resource, 0);
jaroslav@1523
   212
                    return url == null ? null : url.openStream();
jaroslav@1504
   213
                }
jaroslav@1504
   214
            }).generate(sb);
jaroslav@1492
   215
    }
jaroslav@1492
   216
jaroslav@1489
   217
    static class EmulationResources implements Bck2Brwsr.Resources {
jaroslav@1489
   218
jaroslav@1489
   219
        @Override
jaroslav@1489
   220
        public InputStream get(String name) throws IOException {
jaroslav@1489
   221
            Enumeration<URL> en = CompileCP.class.getClassLoader().getResources(name);
jaroslav@1489
   222
            URL u = null;
jaroslav@1489
   223
            while (en.hasMoreElements()) {
jaroslav@1489
   224
                u = en.nextElement();
jaroslav@1489
   225
            }
jaroslav@1489
   226
            if (u == null) {
jaroslav@1523
   227
                LOG.log(Level.WARNING, "Cannot find {0}", name);
jaroslav@1523
   228
                return null;
jaroslav@1489
   229
            }
jaroslav@1523
   230
            if (u.toExternalForm().contains("/rt.jar!")) {
jaroslav@1523
   231
                LOG.warning(name + "No bootdelegation for ");
jaroslav@1523
   232
                return null;
jaroslav@1489
   233
            }
jaroslav@1489
   234
            return u.openStream();
jaroslav@1489
   235
        }
jaroslav@1489
   236
    }
jaroslav@1489
   237
}