launcher/http/src/main/java/org/apidesign/bck2brwsr/launcher/CompileCP.java
author Jaroslav Tulach <jaroslav.tulach@apidesign.org>
Mon, 05 May 2014 10:16:30 +0200
branchclosure
changeset 1525 777bd3ed81ba
parent 1523 d1eeb43a75a3
child 1547 7c10f6d5635c
permissions -rw-r--r--
Export only test classes, classes from exported packages and those referenced in META-INF/services
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> exported = new HashSet<String>();
jaroslav@1525
    53
        Set<String> keep = new HashSet<String>(testClasses);
jaroslav@1525
    54
        listJAR(jar, classes, arr, exported, keep);
jaroslav@1525
    55
        List<String> root = new ArrayList<>();
jaroslav@1525
    56
        for (String c : classes) {
jaroslav@1525
    57
            if (keep.contains(c)) {
jaroslav@1525
    58
                root.add(c);
jaroslav@1525
    59
                continue;
jaroslav@1525
    60
            }
jaroslav@1525
    61
            int slash = c.lastIndexOf('/');
jaroslav@1525
    62
            String pkg = c.substring(0, slash + 1);
jaroslav@1525
    63
            if (exported.contains(pkg)) {
jaroslav@1525
    64
                root.add(c);
jaroslav@1525
    65
            }
jaroslav@1525
    66
        }
jaroslav@1525
    67
        
jaroslav@1489
    68
        StringWriter w = new StringWriter();
jaroslav@1489
    69
        try {
jaroslav@1489
    70
            class JarRes extends EmulationResources implements Bck2Brwsr.Resources {
jaroslav@1489
    71
                @Override
jaroslav@1489
    72
                public InputStream get(String resource) throws IOException {
jaroslav@1489
    73
                    InputStream is = jar.getInputStream(new ZipEntry(resource));
jaroslav@1489
    74
                    return is == null ? super.get(resource) : is;
jaroslav@1489
    75
                }
jaroslav@1489
    76
            }
jaroslav@1489
    77
            
jaroslav@1489
    78
            Bck2Brwsr.newCompiler()
jaroslav@1491
    79
                .addClasses(classes.toArray(new String[0]))
jaroslav@1525
    80
                .addRootClasses(root.toArray(new String[0]))
jaroslav@1498
    81
                .addResources(arr.toArray(new String[0]))
jaroslav@1491
    82
                .library(true)
jaroslav@1489
    83
                .resources(new JarRes())
jaroslav@1489
    84
                .generate(w);
jaroslav@1489
    85
            w.flush();
jaroslav@1489
    86
            return w.toString();
jaroslav@1523
    87
        } catch (IOException ex) {
jaroslav@1523
    88
            throw ex;
jaroslav@1489
    89
        } catch (Throwable ex) {
jaroslav@1489
    90
            throw new IOException("Cannot compile: ", ex);
jaroslav@1489
    91
        } finally {
jaroslav@1489
    92
            w.close();
jaroslav@1489
    93
        }
jaroslav@1489
    94
    }
jaroslav@1489
    95
    
jaroslav@1505
    96
    static String compileFromClassPath(URL u, final Res r) throws IOException {
jaroslav@1505
    97
        File f;
jaroslav@1505
    98
        try {
jaroslav@1505
    99
            f = new File(u.toURI());
jaroslav@1505
   100
        } catch (URISyntaxException ex) {
jaroslav@1505
   101
            throw new IOException(ex);
jaroslav@1505
   102
        }
jaroslav@1489
   103
        for (String s : System.getProperty("java.class.path").split(File.pathSeparator)) {
jaroslav@1489
   104
            if (!f.getPath().startsWith(s)) {
jaroslav@1489
   105
                continue;
jaroslav@1489
   106
            }
jaroslav@1489
   107
            File root = new File(s);
jaroslav@1489
   108
            List<String> arr = new ArrayList<>();
jaroslav@1489
   109
            List<String> classes = new ArrayList<>();
jaroslav@1489
   110
            listDir(root, null, classes, arr);
jaroslav@1489
   111
            StringWriter w = new StringWriter();
jaroslav@1489
   112
            try {
jaroslav@1489
   113
                Bck2Brwsr.newCompiler()
jaroslav@1489
   114
                    .addRootClasses(classes.toArray(new String[0]))
jaroslav@1498
   115
                    .addResources(arr.toArray(new String[0]))
jaroslav@1491
   116
                    .library(true)
jaroslav@1504
   117
                    .resources(new EmulationResources() {
jaroslav@1504
   118
                        @Override
jaroslav@1504
   119
                        public InputStream get(String resource) throws IOException {
jaroslav@1517
   120
                            if (r != null) {
jaroslav@1517
   121
                                final URL url = r.get(resource, 0);
jaroslav@1517
   122
                                return url == null ? null : url.openStream();
jaroslav@1517
   123
                            }
jaroslav@1517
   124
                            return super.get(resource);
jaroslav@1504
   125
                        }
jaroslav@1504
   126
                    })
jaroslav@1489
   127
                    .generate(w);
jaroslav@1489
   128
                w.flush();
jaroslav@1489
   129
                return w.toString();
jaroslav@1489
   130
            } catch (ClassFormatError ex) {
jaroslav@1489
   131
                throw new IOException(ex);
jaroslav@1489
   132
            } finally {
jaroslav@1489
   133
                w.close();
jaroslav@1489
   134
            }
jaroslav@1489
   135
        }
jaroslav@1489
   136
        return null;
jaroslav@1489
   137
    }
jaroslav@1489
   138
    
jaroslav@1525
   139
    private static void listJAR(
jaroslav@1525
   140
        JarFile j, List<String> classes,
jaroslav@1525
   141
        List<String> resources, Set<String> exported, Set<String> keep
jaroslav@1525
   142
    ) throws IOException {
jaroslav@1489
   143
        Enumeration<JarEntry> en = j.entries();
jaroslav@1489
   144
        while (en.hasMoreElements()) {
jaroslav@1489
   145
            JarEntry e = en.nextElement();
jaroslav@1489
   146
            final String n = e.getName();
jaroslav@1492
   147
            if (n.contains("package-info")) {
jaroslav@1492
   148
                continue;
jaroslav@1492
   149
            }
jaroslav@1489
   150
            if (n.endsWith("/")) {
jaroslav@1489
   151
                continue;
jaroslav@1489
   152
            }
jaroslav@1489
   153
            int last = n.lastIndexOf('/');
jaroslav@1489
   154
            String pkg = n.substring(0, last + 1);
jaroslav@1489
   155
            if (skipPkg(pkg)) {
jaroslav@1489
   156
                continue;
jaroslav@1489
   157
            }
jaroslav@1489
   158
            if (n.endsWith(".class")) {
jaroslav@1489
   159
                classes.add(n.substring(0, n.length() - 6));
jaroslav@1489
   160
            } else {
jaroslav@1489
   161
                resources.add(n);
jaroslav@1525
   162
                if (n.startsWith("META-INF/services/") && keep != null) {
jaroslav@1525
   163
                    BufferedReader r = new BufferedReader(new InputStreamReader(j.getInputStream(e)));
jaroslav@1525
   164
                    for (;;) {
jaroslav@1525
   165
                        String l = r.readLine();
jaroslav@1525
   166
                        if (l == null) {
jaroslav@1525
   167
                            break;
jaroslav@1525
   168
                        }
jaroslav@1525
   169
                        if (l.startsWith("#")) {
jaroslav@1525
   170
                            continue;
jaroslav@1525
   171
                        }
jaroslav@1525
   172
                        keep.add(l.replace('.', '/'));
jaroslav@1525
   173
                    }
jaroslav@1525
   174
                }
jaroslav@1525
   175
            }
jaroslav@1525
   176
        }
jaroslav@1525
   177
        String exp = j.getManifest().getMainAttributes().getValue("Export-Package");
jaroslav@1525
   178
        if (exp != null && exported != null) {
jaroslav@1525
   179
            for (String def : exp.split(",")) {
jaroslav@1525
   180
                for (String sep : def.split(";")) {
jaroslav@1525
   181
                    exported.add(sep.replace('.', '/') + "/");
jaroslav@1525
   182
                    break;
jaroslav@1525
   183
                }
jaroslav@1489
   184
            }
jaroslav@1489
   185
        }
jaroslav@1489
   186
    }
jaroslav@1489
   187
jaroslav@1489
   188
    private static boolean skipPkg(String pkg) {
jaroslav@1489
   189
        return pkg.equals("org/apidesign/bck2brwsr/launcher/");
jaroslav@1489
   190
    }
jaroslav@1489
   191
    
jaroslav@1489
   192
    private static void listDir(File f, String pref, List<String> classes, List<String> resources) throws IOException {
jaroslav@1489
   193
        File[] arr = f.listFiles();
jaroslav@1489
   194
        if (arr == null) {
jaroslav@1504
   195
            if (f.getName().equals("package-info.class")) {
jaroslav@1504
   196
                return;
jaroslav@1504
   197
            }
jaroslav@1489
   198
            if (f.getName().endsWith(".class")) {
jaroslav@1489
   199
                classes.add(pref + f.getName().substring(0, f.getName().length() - 6));
jaroslav@1489
   200
            } else {
jaroslav@1489
   201
                resources.add(pref + f.getName());
jaroslav@1489
   202
            }
jaroslav@1489
   203
        } else {
jaroslav@1489
   204
            for (File ch : arr) {
jaroslav@1489
   205
                
jaroslav@1489
   206
                listDir(ch, pref == null ? "" : pref + f.getName() + "/", classes, resources);
jaroslav@1489
   207
            }
jaroslav@1489
   208
        }
jaroslav@1489
   209
    }
jaroslav@1489
   210
jaroslav@1504
   211
    static void compileVM(StringBuilder sb, final Res r) throws IOException {
jaroslav@1513
   212
        URL u = r.get(InterruptedException.class.getName().replace('.', '/') + ".class", 0);
jaroslav@1492
   213
        JarURLConnection juc = (JarURLConnection)u.openConnection();
jaroslav@1492
   214
        
jaroslav@1492
   215
        List<String> arr = new ArrayList<>();
jaroslav@1492
   216
        List<String> classes = new ArrayList<>();
jaroslav@1525
   217
        listJAR(juc.getJarFile(), classes, arr, null, null);
jaroslav@1492
   218
jaroslav@1492
   219
        Bck2Brwsr.newCompiler().addRootClasses(classes.toArray(new String[0]))
jaroslav@1504
   220
            .resources(new Bck2Brwsr.Resources() {
jaroslav@1504
   221
                @Override
jaroslav@1504
   222
                public InputStream get(String resource) throws IOException {
jaroslav@1523
   223
                    final URL url = r.get(resource, 0);
jaroslav@1523
   224
                    return url == null ? null : url.openStream();
jaroslav@1504
   225
                }
jaroslav@1504
   226
            }).generate(sb);
jaroslav@1492
   227
    }
jaroslav@1492
   228
jaroslav@1489
   229
    static class EmulationResources implements Bck2Brwsr.Resources {
jaroslav@1489
   230
jaroslav@1489
   231
        @Override
jaroslav@1489
   232
        public InputStream get(String name) throws IOException {
jaroslav@1489
   233
            Enumeration<URL> en = CompileCP.class.getClassLoader().getResources(name);
jaroslav@1489
   234
            URL u = null;
jaroslav@1489
   235
            while (en.hasMoreElements()) {
jaroslav@1489
   236
                u = en.nextElement();
jaroslav@1489
   237
            }
jaroslav@1489
   238
            if (u == null) {
jaroslav@1523
   239
                LOG.log(Level.WARNING, "Cannot find {0}", name);
jaroslav@1523
   240
                return null;
jaroslav@1489
   241
            }
jaroslav@1523
   242
            if (u.toExternalForm().contains("/rt.jar!")) {
jaroslav@1523
   243
                LOG.warning(name + "No bootdelegation for ");
jaroslav@1523
   244
                return null;
jaroslav@1489
   245
            }
jaroslav@1489
   246
            return u.openStream();
jaroslav@1489
   247
        }
jaroslav@1489
   248
    }
jaroslav@1489
   249
}