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 |
}
|