2 * Back 2 Browser Bytecode Translator
3 * Copyright (C) 2012 Jaroslav Tulach <jaroslav.tulach@apidesign.org>
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.
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.
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.
18 package org.apidesign.bck2brwsr.dew;
20 import java.io.ByteArrayInputStream;
21 import java.io.ByteArrayOutputStream;
22 import java.io.IOException;
23 import java.io.InputStream;
24 import java.io.OutputStream;
26 import java.net.URISyntaxException;
27 import java.util.ArrayList;
28 import java.util.Arrays;
29 import java.util.HashMap;
30 import java.util.List;
32 import javax.tools.Diagnostic;
33 import javax.tools.DiagnosticListener;
34 import javax.tools.FileObject;
35 import javax.tools.ForwardingJavaFileManager;
36 import javax.tools.JavaFileManager;
37 import javax.tools.JavaFileObject;
38 import javax.tools.JavaFileObject.Kind;
39 import javax.tools.SimpleJavaFileObject;
40 import javax.tools.StandardJavaFileManager;
41 import javax.tools.StandardLocation;
42 import javax.tools.ToolProvider;
46 * @author Jaroslav Tulach <jtulach@netbeans.org>
48 final class Compile implements DiagnosticListener<JavaFileObject> {
49 private final List<Diagnostic<? extends JavaFileObject>> errors = new ArrayList<>();
50 private final Map<String, byte[]> classes;
51 private final String pkg;
52 private final String cls;
53 private final String html;
55 private Compile(String html, String code) throws IOException {
56 this.pkg = find("package", ';', code);
57 this.cls = find("class", ' ', code);
59 classes = compile(html, code);
62 /** Performs compilation of given HTML page and associated Java code
64 public static Compile create(String html, String code) throws IOException {
65 return new Compile(html, code);
68 /** Checks for given class among compiled resources */
69 public byte[] get(String res) {
70 return classes.get(res);
73 /** Obtains errors created during compilation.
75 public List<Diagnostic<? extends JavaFileObject>> getErrors() {
76 List<Diagnostic<? extends JavaFileObject>> err = new ArrayList<>();
77 for (Diagnostic<? extends JavaFileObject> diagnostic : errors) {
78 if (diagnostic.getKind() == Diagnostic.Kind.ERROR) {
85 private Map<String, byte[]> compile(final String html, final String code) throws IOException {
86 StandardJavaFileManager sjfm = ToolProvider.getSystemJavaCompiler().getStandardFileManager(this, null, null);
88 final Map<String, ByteArrayOutputStream> class2BAOS = new HashMap<>();
90 JavaFileObject file = new SimpleJavaFileObject(URI.create("mem://mem"), Kind.SOURCE) {
92 public CharSequence getCharContent(boolean ignoreEncodingErrors) throws IOException {
96 final JavaFileObject htmlFile = new SimpleJavaFileObject(URI.create("mem://mem2"), Kind.OTHER) {
98 public CharSequence getCharContent(boolean ignoreEncodingErrors) throws IOException {
103 public InputStream openInputStream() throws IOException {
104 return new ByteArrayInputStream(html.getBytes());
110 scratch = new URI("mem://mem3");
111 } catch (URISyntaxException ex) {
112 throw new IOException(ex);
115 JavaFileManager jfm = new ForwardingJavaFileManager<JavaFileManager>(sjfm) {
117 public JavaFileObject getJavaFileForOutput(Location location, String className, Kind kind, FileObject sibling) throws IOException {
118 if (kind == Kind.CLASS) {
119 final ByteArrayOutputStream buffer = new ByteArrayOutputStream();
121 class2BAOS.put(className.replace('.', '/') + ".class", buffer);
122 return new SimpleJavaFileObject(sibling.toUri(), kind) {
124 public OutputStream openOutputStream() throws IOException {
130 if (kind == Kind.SOURCE) {
131 return new SimpleJavaFileObject(scratch/*sibling.toUri()*/, kind) {
132 private final ByteArrayOutputStream data = new ByteArrayOutputStream();
134 public OutputStream openOutputStream() throws IOException {
139 public CharSequence getCharContent(boolean ignoreEncodingErrors) throws IOException {
141 return new String(data.toByteArray());
146 throw new IllegalStateException();
150 public FileObject getFileForInput(Location location, String packageName, String relativeName) throws IOException {
151 if (location == StandardLocation.SOURCE_PATH) {
152 if (packageName.equals(pkg)) {
162 ToolProvider.getSystemJavaCompiler().getTask(null, jfm, this, /*XXX:*/Arrays.asList("-source", "1.7", "-target", "1.7"), null, Arrays.asList(file)).call();
164 Map<String, byte[]> result = new HashMap<>();
166 for (Map.Entry<String, ByteArrayOutputStream> e : class2BAOS.entrySet()) {
167 result.put(e.getKey(), e.getValue().toByteArray());
175 public void report(Diagnostic<? extends JavaFileObject> diagnostic) {
176 errors.add(diagnostic);
178 private static String find(String pref, char term, String java) throws IOException {
179 int pkg = java.indexOf(pref);
181 pkg += pref.length();
182 while (Character.isWhitespace(java.charAt(pkg))) {
185 int semicolon = java.indexOf(term, pkg);
186 if (semicolon != -1) {
187 String pkgName = java.substring(pkg, semicolon).trim();
191 throw new IOException("Can't find " + pref + " declaration in the java file");
195 String fqn = "'" + pkg + '.' + cls + "'";
196 return html.replace("'${fqn}'", fqn);