Add source copying in maven plugin:

https://github.com/konsoletyper/teavm/issues/13
This commit is contained in:
Alexey Andreev 2014-09-26 18:52:28 +04:00
parent 15299213d2
commit 6c05ec589a
16 changed files with 581 additions and 7 deletions

View File

@ -137,6 +137,11 @@
<artifactId>maven-core</artifactId> <artifactId>maven-core</artifactId>
<version>3.0</version> <version>3.0</version>
</dependency> </dependency>
<dependency>
<groupId>org.apache.maven</groupId>
<artifactId>maven-artifact</artifactId>
<version>3.0</version>
</dependency>
<dependency> <dependency>
<groupId>javax.websocket</groupId> <groupId>javax.websocket</groupId>
<artifactId>javax.websocket-api</artifactId> <artifactId>javax.websocket-api</artifactId>

View File

@ -377,8 +377,8 @@ public class DebugInformation {
writer.write(this); writer.write(this);
} }
public void writeAsSourceMaps(Writer output, String sourceFile) throws IOException { public void writeAsSourceMaps(Writer output, String sourceRoot, String sourceFile) throws IOException {
new SourceMapsWriter(output).write(sourceFile, this); new SourceMapsWriter(output).write(sourceFile, sourceRoot, this);
} }
public static DebugInformation read(InputStream input) throws IOException { public static DebugInformation read(InputStream input) throws IOException {

View File

@ -35,12 +35,14 @@ class SourceMapsWriter {
this.output = output; this.output = output;
} }
public void write(String generatedFile, DebugInformation debugInfo) throws IOException { public void write(String generatedFile, String sourceRoot, DebugInformation debugInfo) throws IOException {
output.write("{\"version\":3"); output.write("{\"version\":3");
output.write(",\"file\":\""); output.write(",\"file\":\"");
writeEscapedString(generatedFile); writeEscapedString(generatedFile);
output.write("\""); output.write("\"");
output.write(",\"sourceRoot\":\"\""); output.write(",\"sourceRoot\":\"");
writeEscapedString(sourceRoot);
output.write("\"");
output.write(",\"sources\":["); output.write(",\"sources\":[");
for (int i = 0; i < debugInfo.fileNames.length; ++i) { for (int i = 0; i < debugInfo.fileNames.length; ++i) {
if (i > 0) { if (i > 0) {
@ -59,7 +61,7 @@ class SourceMapsWriter {
lastSourceFile = 0; lastSourceFile = 0;
lastSourceLine = 0; lastSourceLine = 0;
for (SourceLocationIterator iter = debugInfo.iterateOverSourceLocations(); !iter.isEndReached(); iter.next()) { for (SourceLocationIterator iter = debugInfo.iterateOverSourceLocations(); !iter.isEndReached(); iter.next()) {
writeSegment(iter.getLocation(), iter.getFileNameId(), iter.getLine()); writeSegment(iter.getLocation(), iter.getFileNameId(), iter.getLine() - 1);
} }
output.write("\"}"); output.write("\"}");
} }

View File

@ -0,0 +1,48 @@
/*
* Copyright 2014 Alexey Andreev.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.teavm.tooling;
import java.io.File;
import java.io.FileInputStream;
import java.io.IOException;
import java.io.InputStream;
/**
*
* @author Alexey Andreev
*/
public class DirectorySourceFileProvider implements SourceFileProvider {
private File baseDirectory;
public DirectorySourceFileProvider(File baseDirectory) {
this.baseDirectory = baseDirectory;
}
@Override
public void open() throws IOException {
}
@Override
public void close() throws IOException {
}
@Override
@SuppressWarnings("resource")
public InputStream openSourceFile(String fullPath) throws IOException {
File file = new File(baseDirectory, fullPath);
return file.exists() ? new FileInputStream(file) : null;
}
}

View File

@ -0,0 +1,59 @@
/*
* Copyright 2014 Alexey Andreev.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.teavm.tooling;
import java.io.File;
import java.io.IOException;
import java.io.InputStream;
import java.util.zip.ZipEntry;
import java.util.zip.ZipFile;
/**
*
* @author Alexey Andreev
*/
public class JarSourceFileProvider implements SourceFileProvider {
private File file;
private ZipFile zipFile;
public JarSourceFileProvider(File file) {
this.file = file;
}
@Override
public void open() throws IOException {
zipFile = new ZipFile(file);
}
@Override
public void close() throws IOException {
if (this.zipFile == null) {
return;
}
ZipFile zipFile = this.zipFile;
this.zipFile = null;
zipFile.close();
}
@Override
public InputStream openSourceFile(String fullPath) throws IOException {
if (zipFile == null) {
return null;
}
ZipEntry entry = zipFile.getEntry(fullPath);
return entry != null ? zipFile.getInputStream(entry) : null;
}
}

View File

@ -0,0 +1,91 @@
/*
* Copyright 2014 Alexey Andreev.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.teavm.tooling;
import java.util.List;
import java.util.Set;
import org.teavm.model.*;
import org.teavm.model.instructions.*;
/**
*
* @author Alexey Andreev
*/
class ProgramSourceAggregator implements InstructionReader {
private Set<String> sourceFiles;
public ProgramSourceAggregator(Set<String> sourceFiles) {
this.sourceFiles = sourceFiles;
}
public void addLocationsOfProgram(ProgramReader program) {
for (int i = 0; i < program.basicBlockCount(); ++i) {
BasicBlockReader block = program.basicBlockAt(i);
block.readAllInstructions(this);
}
}
@Override
public void location(InstructionLocation location) {
if (location != null && location.getFileName() != null && !location.getFileName().isEmpty()) {
sourceFiles.add(location.getFileName());
}
}
@Override public void nop() { }
@Override public void classConstant(VariableReader receiver, ValueType cst) { }
@Override public void nullConstant(VariableReader receiver) { }
@Override public void integerConstant(VariableReader receiver, int cst) { }
@Override public void longConstant(VariableReader receiver, long cst) { }
@Override public void floatConstant(VariableReader receiver, float cst) { }
@Override public void doubleConstant(VariableReader receiver, double cst) { }
@Override public void stringConstant(VariableReader receiver, String cst) { }
@Override public void binary(BinaryOperation op, VariableReader receiver, VariableReader first,
VariableReader second, NumericOperandType type) { }
@Override public void negate(VariableReader receiver, VariableReader operand, NumericOperandType type) { }
@Override public void assign(VariableReader receiver, VariableReader assignee) { }
@Override public void cast(VariableReader receiver, VariableReader value, ValueType targetType) { }
@Override public void cast(VariableReader receiver, VariableReader value, NumericOperandType sourceType,
NumericOperandType targetType) { }
@Override public void cast(VariableReader receiver, VariableReader value, IntegerSubtype type,
CastIntegerDirection targetType) { }
@Override public void jumpIf(BranchingCondition cond, VariableReader operand, BasicBlockReader consequent,
BasicBlockReader alternative) { }
@Override public void jumpIf(BinaryBranchingCondition cond, VariableReader first, VariableReader second,
BasicBlockReader consequent, BasicBlockReader alternative) { }
@Override public void jump(BasicBlockReader target) { }
@Override public void choose(VariableReader condition, List<? extends SwitchTableEntryReader> table,
BasicBlockReader defaultTarget) { }
@Override public void exit(VariableReader valueToReturn) { }
@Override public void raise(VariableReader exception) { }
@Override public void createArray(VariableReader receiver, ValueType itemType, VariableReader size) { }
@Override public void createArray(VariableReader receiver, ValueType itemType,
List<? extends VariableReader> dimensions) { }
@Override public void create(VariableReader receiver, String type) { }
@Override public void getField(VariableReader receiver, VariableReader instance, FieldReference field,
ValueType fieldType) { }
@Override public void putField(VariableReader instance, FieldReference field, VariableReader value) { }
@Override public void arrayLength(VariableReader receiver, VariableReader array) { }
@Override public void cloneArray(VariableReader receiver, VariableReader array) { }
@Override public void unwrapArray(VariableReader receiver, VariableReader array, ArrayElementType elementType) { }
@Override public void getElement(VariableReader receiver, VariableReader array, VariableReader index) { }
@Override public void putElement(VariableReader array, VariableReader index, VariableReader value) { }
@Override public void invoke(VariableReader receiver, VariableReader instance, MethodReference method,
List<? extends VariableReader> arguments, InvocationType type) { }
@Override public void isInstance(VariableReader receiver, VariableReader value, ValueType type) { }
@Override public void initClass(String className) { }
@Override public void nullCheck(VariableReader receiver, VariableReader value) { }
}

View File

@ -0,0 +1,31 @@
/*
* Copyright 2014 Alexey Andreev.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.teavm.tooling;
import java.io.IOException;
import java.io.InputStream;
/**
*
* @author Alexey Andreev
*/
public interface SourceFileProvider {
void open() throws IOException;
void close() throws IOException;
InputStream openSourceFile(String fullPath) throws IOException;
}

View File

@ -0,0 +1,99 @@
/*
* Copyright 2014 Alexey Andreev.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.teavm.tooling;
import java.io.*;
import java.util.HashSet;
import java.util.List;
import java.util.Set;
import org.apache.commons.io.IOUtils;
import org.teavm.model.ClassReader;
import org.teavm.model.ListableClassReaderSource;
import org.teavm.model.MethodReader;
/**
*
* @author Alexey Andreev
*/
public class SourceFilesCopier {
private TeaVMToolLog log = new EmptyTeaVMToolLog();
private List<SourceFileProvider> sourceFileProviders;
private Set<String> sourceFiles = new HashSet<>();
public SourceFilesCopier(List<SourceFileProvider> sourceFileProviders) {
this.sourceFileProviders = sourceFileProviders;
}
public void setLog(TeaVMToolLog log) {
this.log = log;
}
public void addClasses(ListableClassReaderSource classReader) {
ProgramSourceAggregator sourceAggregator = new ProgramSourceAggregator(sourceFiles);
for (String className : classReader.getClassNames()) {
ClassReader cls = classReader.get(className);
for (MethodReader method : cls.getMethods()) {
if (method.getProgram() == null) {
continue;
}
sourceAggregator.addLocationsOfProgram(method.getProgram());
}
}
}
public void copy(File targetDirectory) {
for (SourceFileProvider provider : sourceFileProviders) {
try {
provider.open();
} catch (IOException e) {
log.warning("Error opening source file provider", e);
}
}
targetDirectory.mkdirs();
for (String fileName : sourceFiles) {
try (InputStream input = findSourceFile(fileName)) {
if (input != null) {
File outputFile = new File(targetDirectory, fileName);
outputFile.getParentFile().mkdirs();
try (OutputStream output = new FileOutputStream(outputFile)) {
IOUtils.copy(input, output);
}
} else {
log.info("Missing source file: " + fileName);
}
} catch (IOException e) {
log.warning("Could not copy source file " + fileName, e);
}
}
for (SourceFileProvider provider : sourceFileProviders) {
try {
provider.close();
} catch (IOException e) {
log.warning("Error closing source file provider", e);
}
}
}
private InputStream findSourceFile(String fileName) throws IOException {
for (SourceFileProvider provider : sourceFileProviders) {
InputStream input = provider.openSourceFile(fileName);
if (input != null) {
return input;
}
}
return null;
}
}

View File

@ -55,9 +55,12 @@ public class TeaVMTestTool {
private TeaVMToolLog log = new EmptyTeaVMToolLog(); private TeaVMToolLog log = new EmptyTeaVMToolLog();
private boolean debugInformationGenerated; private boolean debugInformationGenerated;
private boolean sourceMapsGenerated; private boolean sourceMapsGenerated;
private boolean sourceFilesCopied;
private boolean incremental; private boolean incremental;
private List<SourceFileProvider> sourceFileProviders = new ArrayList<>();
private RegularMethodNodeCache astCache; private RegularMethodNodeCache astCache;
private ProgramCache programCache; private ProgramCache programCache;
private SourceFilesCopier sourceFilesCopier;
public File getOutputDir() { public File getOutputDir() {
return outputDir; return outputDir;
@ -147,6 +150,18 @@ public class TeaVMTestTool {
this.sourceMapsGenerated = sourceMapsGenerated; this.sourceMapsGenerated = sourceMapsGenerated;
} }
public boolean isSourceFilesCopied() {
return sourceFilesCopied;
}
public void setSourceFilesCopied(boolean sourceFilesCopied) {
this.sourceFilesCopied = sourceFilesCopied;
}
public void addSourceFileProvider(SourceFileProvider sourceFileProvider) {
sourceFileProviders.add(sourceFileProvider);
}
public void generate() throws TeaVMToolException { public void generate() throws TeaVMToolException {
Runnable finalizer = null; Runnable finalizer = null;
try { try {
@ -233,6 +248,8 @@ public class TeaVMTestTool {
} }
int methodsGenerated = 0; int methodsGenerated = 0;
log.info("Generating test files"); log.info("Generating test files");
sourceFilesCopier = new SourceFilesCopier(sourceFileProviders);
sourceFilesCopier.setLog(log);
FiniteExecutor executor = new SimpleFiniteExecutor(); FiniteExecutor executor = new SimpleFiniteExecutor();
if (numThreads != 1) { if (numThreads != 1) {
int threads = numThreads != 0 ? numThreads : Runtime.getRuntime().availableProcessors(); int threads = numThreads != 0 ? numThreads : Runtime.getRuntime().availableProcessors();
@ -260,6 +277,9 @@ public class TeaVMTestTool {
++methodsGenerated; ++methodsGenerated;
} }
executor.complete(); executor.complete();
if (sourceFilesCopied) {
sourceFilesCopier.copy(new File(new File(outputDir, "tests"), "src"));
}
log.info("Test files successfully generated for " + methodsGenerated + " method(s)."); log.info("Test files successfully generated for " + methodsGenerated + " method(s).");
} catch (IOException e) { } catch (IOException e) {
throw new TeaVMToolException("IO error occured generating JavaScript files", e); throw new TeaVMToolException("IO error occured generating JavaScript files", e);
@ -374,9 +394,12 @@ public class TeaVMTestTool {
String sourceMapsFileName = targetName + ".map"; String sourceMapsFileName = targetName + ".map";
try (Writer sourceMapsOut = new OutputStreamWriter(new FileOutputStream( try (Writer sourceMapsOut = new OutputStreamWriter(new FileOutputStream(
new File(outputDir, sourceMapsFileName)), "UTF-8")) { new File(outputDir, sourceMapsFileName)), "UTF-8")) {
debugInfo.writeAsSourceMaps(sourceMapsOut, targetName); debugInfo.writeAsSourceMaps(sourceMapsOut, "src", targetName);
} }
} }
if (sourceFilesCopied && vm.getWrittenClasses() != null) {
sourceFilesCopier.addClasses(vm.getWrittenClasses());
}
} }
private void escapeStringLiteral(String text, Writer writer) throws IOException { private void escapeStringLiteral(String text, Writer writer) throws IOException {

View File

@ -46,6 +46,7 @@ public class TeaVMTool {
private boolean bytecodeLogging; private boolean bytecodeLogging;
private boolean debugInformationGenerated; private boolean debugInformationGenerated;
private boolean sourceMapsFileGenerated; private boolean sourceMapsFileGenerated;
private boolean sourceFilesCopied;
private boolean incremental; private boolean incremental;
private File cacheDirectory = new File("./teavm-cache"); private File cacheDirectory = new File("./teavm-cache");
private List<ClassHolderTransformer> transformers = new ArrayList<>(); private List<ClassHolderTransformer> transformers = new ArrayList<>();
@ -61,6 +62,7 @@ public class TeaVMTool {
private boolean cancelled; private boolean cancelled;
private TeaVMProgressListener progressListener; private TeaVMProgressListener progressListener;
private TeaVM vm; private TeaVM vm;
private List<SourceFileProvider> sourceFileProviders = new ArrayList<>();
public File getTargetDirectory() { public File getTargetDirectory() {
return targetDirectory; return targetDirectory;
@ -150,6 +152,14 @@ public class TeaVMTool {
this.sourceMapsFileGenerated = sourceMapsFileGenerated; this.sourceMapsFileGenerated = sourceMapsFileGenerated;
} }
public boolean isSourceFilesCopied() {
return sourceFilesCopied;
}
public void setSourceFilesCopied(boolean sourceFilesCopied) {
this.sourceFilesCopied = sourceFilesCopied;
}
public Properties getProperties() { public Properties getProperties() {
return properties; return properties;
} }
@ -194,6 +204,10 @@ public class TeaVMTool {
return vm != null ? vm.getClasses() : Collections.<String>emptyList(); return vm != null ? vm.getClasses() : Collections.<String>emptyList();
} }
public void addSourceFileProvider(SourceFileProvider sourceFileProvider) {
sourceFileProviders.add(sourceFileProvider);
}
public void generate() throws TeaVMToolException { public void generate() throws TeaVMToolException {
try { try {
cancelled = false; cancelled = false;
@ -295,10 +309,14 @@ public class TeaVMTool {
writer.append("\n//# sourceMappingURL=").append(sourceMapsFileName); writer.append("\n//# sourceMappingURL=").append(sourceMapsFileName);
try (Writer sourceMapsOut = new OutputStreamWriter(new FileOutputStream( try (Writer sourceMapsOut = new OutputStreamWriter(new FileOutputStream(
new File(targetDirectory, sourceMapsFileName)), "UTF-8")) { new File(targetDirectory, sourceMapsFileName)), "UTF-8")) {
debugInfo.writeAsSourceMaps(sourceMapsOut, targetFileName); debugInfo.writeAsSourceMaps(sourceMapsOut, "src", targetFileName);
} }
log.info("Source maps successfully written"); log.info("Source maps successfully written");
} }
if (sourceFilesCopied) {
copySourceFiles();
log.info("Source files successfully written");
}
if (incremental) { if (incremental) {
programCache.flush(); programCache.flush();
astCache.flush(); astCache.flush();
@ -335,6 +353,16 @@ public class TeaVMTool {
vm.checkForMissingItems(); vm.checkForMissingItems();
} }
private void copySourceFiles() {
if (vm.getWrittenClasses() == null) {
return;
}
SourceFilesCopier copier = new SourceFilesCopier(sourceFileProviders);
copier.addClasses(vm.getWrittenClasses());
copier.setLog(log);
copier.copy(new File(targetDirectory, "src"));
}
private AbstractRendererListener runtimeInjector = new AbstractRendererListener() { private AbstractRendererListener runtimeInjector = new AbstractRendererListener() {
@Override @Override
public void begin(RenderingContext context, BuildTarget buildTarget) throws IOException { public void begin(RenderingContext context, BuildTarget buildTarget) throws IOException {

View File

@ -85,6 +85,7 @@ public class TeaVM implements TeaVMHost, ServiceRepository {
private boolean incremental; private boolean incremental;
private TeaVMProgressListener progressListener; private TeaVMProgressListener progressListener;
private boolean cancelled; private boolean cancelled;
private ListableClassHolderSource writtenClasses;
TeaVM(ClassReaderSource classSource, ClassLoader classLoader) { TeaVM(ClassReaderSource classSource, ClassLoader classLoader) {
this.classSource = classSource; this.classSource = classSource;
@ -312,6 +313,14 @@ public class TeaVM implements TeaVMHost, ServiceRepository {
return dependencyChecker.getAchievableClasses(); return dependencyChecker.getAchievableClasses();
} }
public DependencyInfo getDependencyInfo() {
return dependencyChecker;
}
public ListableClassReaderSource getWrittenClasses() {
return writtenClasses;
}
/** /**
* <p>After building checks whether the build has failed due to some missing items (classes, methods and fields). * <p>After building checks whether the build has failed due to some missing items (classes, methods and fields).
* If it has failed, throws exception, containing report on all missing items. * If it has failed, throws exception, containing report on all missing items.
@ -378,6 +387,7 @@ public class TeaVM implements TeaVMHost, ServiceRepository {
return; return;
} }
ListableClassHolderSource classSet = link(dependencyChecker); ListableClassHolderSource classSet = link(dependencyChecker);
writtenClasses = classSet;
if (wasCancelled()) { if (wasCancelled()) {
return; return;
} }

View File

@ -43,6 +43,11 @@
<artifactId>maven-core</artifactId> <artifactId>maven-core</artifactId>
<scope>provided</scope> <scope>provided</scope>
</dependency> </dependency>
<dependency>
<groupId>org.apache.maven</groupId>
<artifactId>maven-artifact</artifactId>
<scope>provided</scope>
</dependency>
<dependency> <dependency>
<groupId>org.teavm</groupId> <groupId>org.teavm</groupId>
<artifactId>teavm-core</artifactId> <artifactId>teavm-core</artifactId>

View File

@ -23,6 +23,7 @@ import java.net.URL;
import java.net.URLClassLoader; import java.net.URLClassLoader;
import java.util.*; import java.util.*;
import org.apache.maven.artifact.Artifact; import org.apache.maven.artifact.Artifact;
import org.apache.maven.artifact.repository.MavenArtifactRepository;
import org.apache.maven.plugin.AbstractMojo; import org.apache.maven.plugin.AbstractMojo;
import org.apache.maven.plugin.MojoExecutionException; import org.apache.maven.plugin.MojoExecutionException;
import org.apache.maven.plugin.logging.Log; import org.apache.maven.plugin.logging.Log;
@ -31,6 +32,7 @@ import org.apache.maven.plugins.annotations.Mojo;
import org.apache.maven.plugins.annotations.Parameter; import org.apache.maven.plugins.annotations.Parameter;
import org.apache.maven.plugins.annotations.ResolutionScope; import org.apache.maven.plugins.annotations.ResolutionScope;
import org.apache.maven.project.MavenProject; import org.apache.maven.project.MavenProject;
import org.apache.maven.repository.RepositorySystem;
import org.teavm.model.ClassHolderTransformer; import org.teavm.model.ClassHolderTransformer;
import org.teavm.tooling.*; import org.teavm.tooling.*;
@ -47,6 +49,18 @@ public class BuildJavascriptMojo extends AbstractMojo {
@Component @Component
private MavenProject project; private MavenProject project;
@Component
private RepositorySystem repositorySystem;
@Parameter(required = true, readonly = true, defaultValue = "${localRepository}")
private MavenArtifactRepository localRepository;
@Parameter(required = true, readonly = true, defaultValue = "${project.remoteArtifactRepositories}")
private List<MavenArtifactRepository> remoteRepositories;
@Parameter(readonly = true, defaultValue = "${plugin.artifacts}")
private List<Artifact> pluginArtifacts;
@Parameter(defaultValue = "${project.build.directory}/javascript") @Parameter(defaultValue = "${project.build.directory}/javascript")
private File targetDirectory; private File targetDirectory;
@ -80,6 +94,9 @@ public class BuildJavascriptMojo extends AbstractMojo {
@Parameter @Parameter
private boolean sourceMapsGenerated; private boolean sourceMapsGenerated;
@Parameter
private boolean sourceFilesCopied;
@Parameter @Parameter
private boolean incremental; private boolean incremental;
@ -165,6 +182,14 @@ public class BuildJavascriptMojo extends AbstractMojo {
this.sourceMapsGenerated = sourceMapsGenerated; this.sourceMapsGenerated = sourceMapsGenerated;
} }
public boolean isSourceFilesCopied() {
return sourceFilesCopied;
}
public void setSourceFilesCopied(boolean sourceFilesCopied) {
this.sourceFilesCopied = sourceFilesCopied;
}
public boolean isIncremental() { public boolean isIncremental() {
return incremental; return incremental;
} }
@ -196,6 +221,17 @@ public class BuildJavascriptMojo extends AbstractMojo {
tool.setTargetDirectory(targetDirectory); tool.setTargetDirectory(targetDirectory);
tool.setTargetFileName(targetFileName); tool.setTargetFileName(targetFileName);
tool.getTransformers().addAll(instantiateTransformers(classLoader)); tool.getTransformers().addAll(instantiateTransformers(classLoader));
if (sourceFilesCopied) {
MavenSourceFileProviderLookup lookup = new MavenSourceFileProviderLookup();
lookup.setMavenProject(project);
lookup.setRepositorySystem(repositorySystem);
lookup.setLocalRepository(localRepository);
lookup.setRemoteRepositories(remoteRepositories);
lookup.setPluginDependencies(pluginArtifacts);
for (SourceFileProvider provider : lookup.resolve()) {
tool.addSourceFileProvider(provider);
}
}
if (classAliases != null) { if (classAliases != null) {
tool.getClassAliases().addAll(Arrays.asList(classAliases)); tool.getClassAliases().addAll(Arrays.asList(classAliases));
} }
@ -209,6 +245,7 @@ public class BuildJavascriptMojo extends AbstractMojo {
tool.setIncremental(incremental); tool.setIncremental(incremental);
tool.setDebugInformationGenerated(debugInformationGenerated); tool.setDebugInformationGenerated(debugInformationGenerated);
tool.setSourceMapsFileGenerated(sourceMapsGenerated); tool.setSourceMapsFileGenerated(sourceMapsGenerated);
tool.setSourceFilesCopied(sourceFilesCopied);
tool.generate(); tool.generate();
tool.checkForMissingItems(); tool.checkForMissingItems();
} catch (RuntimeException e) { } catch (RuntimeException e) {

View File

@ -27,6 +27,7 @@ import java.util.jar.JarEntry;
import java.util.jar.JarFile; import java.util.jar.JarFile;
import org.apache.commons.io.FilenameUtils; import org.apache.commons.io.FilenameUtils;
import org.apache.maven.artifact.Artifact; import org.apache.maven.artifact.Artifact;
import org.apache.maven.artifact.repository.MavenArtifactRepository;
import org.apache.maven.plugin.AbstractMojo; import org.apache.maven.plugin.AbstractMojo;
import org.apache.maven.plugin.MojoExecutionException; import org.apache.maven.plugin.MojoExecutionException;
import org.apache.maven.plugin.MojoFailureException; import org.apache.maven.plugin.MojoFailureException;
@ -36,9 +37,11 @@ import org.apache.maven.plugins.annotations.Mojo;
import org.apache.maven.plugins.annotations.Parameter; import org.apache.maven.plugins.annotations.Parameter;
import org.apache.maven.plugins.annotations.ResolutionScope; import org.apache.maven.plugins.annotations.ResolutionScope;
import org.apache.maven.project.MavenProject; import org.apache.maven.project.MavenProject;
import org.apache.maven.repository.RepositorySystem;
import org.teavm.model.ClassHolderTransformer; import org.teavm.model.ClassHolderTransformer;
import org.teavm.testing.JUnitTestAdapter; import org.teavm.testing.JUnitTestAdapter;
import org.teavm.testing.TestAdapter; import org.teavm.testing.TestAdapter;
import org.teavm.tooling.SourceFileProvider;
import org.teavm.tooling.TeaVMTestTool; import org.teavm.tooling.TeaVMTestTool;
import org.teavm.tooling.TeaVMToolException; import org.teavm.tooling.TeaVMToolException;
@ -56,6 +59,18 @@ public class BuildJavascriptTestMojo extends AbstractMojo {
@Component @Component
private MavenProject project; private MavenProject project;
@Component
private RepositorySystem repositorySystem;
@Parameter(required = true, readonly = true, defaultValue = "${localRepository}")
private MavenArtifactRepository localRepository;
@Parameter(required = true, readonly = true, defaultValue = "${project.remoteArtifactRepositories}")
private List<MavenArtifactRepository> remoteRepositories;
@Parameter(readonly = true, defaultValue = "${plugin.artifacts}")
private List<Artifact> pluginArtifacts;
@Parameter(defaultValue = "${project.build.directory}/javascript-test") @Parameter(defaultValue = "${project.build.directory}/javascript-test")
private File outputDir; private File outputDir;
@ -98,6 +113,9 @@ public class BuildJavascriptTestMojo extends AbstractMojo {
@Parameter @Parameter
private boolean sourceMapsGenerated; private boolean sourceMapsGenerated;
@Parameter
private boolean sourceFilesCopied;
private TeaVMTestTool tool = new TeaVMTestTool(); private TeaVMTestTool tool = new TeaVMTestTool();
public void setProject(MavenProject project) { public void setProject(MavenProject project) {
@ -164,6 +182,14 @@ public class BuildJavascriptTestMojo extends AbstractMojo {
this.sourceMapsGenerated = sourceMapsGenerated; this.sourceMapsGenerated = sourceMapsGenerated;
} }
public boolean isSourceFilesCopied() {
return sourceFilesCopied;
}
public void setSourceFilesCopied(boolean sourceFilesCopied) {
this.sourceFilesCopied = sourceFilesCopied;
}
@Override @Override
public void execute() throws MojoExecutionException, MojoFailureException { public void execute() throws MojoExecutionException, MojoFailureException {
if (System.getProperty("maven.test.skip", "false").equals("true") || if (System.getProperty("maven.test.skip", "false").equals("true") ||
@ -188,6 +214,18 @@ public class BuildJavascriptTestMojo extends AbstractMojo {
tool.setIncremental(incremental); tool.setIncremental(incremental);
tool.setDebugInformationGenerated(debugInformationGenerated); tool.setDebugInformationGenerated(debugInformationGenerated);
tool.setSourceMapsGenerated(sourceMapsGenerated); tool.setSourceMapsGenerated(sourceMapsGenerated);
tool.setSourceFilesCopied(sourceFilesCopied);
if (sourceFilesCopied) {
MavenSourceFileProviderLookup lookup = new MavenSourceFileProviderLookup();
lookup.setMavenProject(project);
lookup.setRepositorySystem(repositorySystem);
lookup.setLocalRepository(localRepository);
lookup.setRemoteRepositories(remoteRepositories);
lookup.setPluginDependencies(pluginArtifacts);
for (SourceFileProvider provider : lookup.resolve()) {
tool.addSourceFileProvider(provider);
}
}
if (properties != null) { if (properties != null) {
tool.getProperties().putAll(properties); tool.getProperties().putAll(properties);
} }

View File

@ -0,0 +1,97 @@
/*
* Copyright 2013 Alexey Andreev.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.teavm.maven;
import java.io.File;
import java.util.ArrayList;
import java.util.HashSet;
import java.util.List;
import java.util.Set;
import org.apache.maven.artifact.Artifact;
import org.apache.maven.artifact.repository.ArtifactRepository;
import org.apache.maven.artifact.resolver.ArtifactResolutionRequest;
import org.apache.maven.artifact.resolver.ArtifactResolutionResult;
import org.apache.maven.project.MavenProject;
import org.apache.maven.repository.RepositorySystem;
import org.teavm.tooling.DirectorySourceFileProvider;
import org.teavm.tooling.JarSourceFileProvider;
import org.teavm.tooling.SourceFileProvider;
/**
*
* @author Alexey Andreev
*/
public class MavenSourceFileProviderLookup {
private MavenProject mavenProject;
private RepositorySystem repositorySystem;
private ArtifactRepository localRepository;
private List<? extends ArtifactRepository> remoteRepositories;
private List<Artifact> pluginDependencies;
public void setMavenProject(MavenProject mavenProject) {
this.mavenProject = mavenProject;
}
public void setRepositorySystem(RepositorySystem repositorySystem) {
this.repositorySystem = repositorySystem;
}
public void setLocalRepository(ArtifactRepository localRepository) {
this.localRepository = localRepository;
}
public void setRemoteRepositories(List<? extends ArtifactRepository> remoteRepositories) {
this.remoteRepositories = remoteRepositories;
}
public void setPluginDependencies(List<Artifact> pluginDependencies) {
this.pluginDependencies = pluginDependencies;
}
public List<SourceFileProvider> resolve() {
List<Artifact> initialArtifacts = new ArrayList<>();
initialArtifacts.addAll(mavenProject.getArtifacts());
if (pluginDependencies != null) {
initialArtifacts.addAll(pluginDependencies);
}
Set<Artifact> artifacts = new HashSet<>();
for (Artifact artifact : initialArtifacts) {
if (artifact.getClassifier() != null && artifact.getClassifier().equals("sources")) {
artifacts.add(artifact);
} else {
artifacts.add(repositorySystem.createArtifactWithClassifier(artifact.getGroupId(),
artifact.getArtifactId(), artifact.getVersion(), artifact.getType(), "sources"));
}
}
List<SourceFileProvider> providers = new ArrayList<>();
for (Artifact artifact : artifacts) {
ArtifactResolutionRequest request = new ArtifactResolutionRequest()
.setLocalRepository(localRepository)
.setRemoteRepositories(new ArrayList<>(remoteRepositories))
.setArtifact(artifact);
ArtifactResolutionResult result = repositorySystem.resolve(request);
for (Artifact resolvedArtifact : result.getArtifacts()) {
if (resolvedArtifact.getFile() != null) {
providers.add(new JarSourceFileProvider(resolvedArtifact.getFile()));
}
}
}
for (String sourceRoot : mavenProject.getCompileSourceRoots()) {
providers.add(new DirectorySourceFileProvider(new File(sourceRoot)));
}
return providers;
}
}

View File

@ -69,6 +69,7 @@
<mainPageIncluded>true</mainPageIncluded> <mainPageIncluded>true</mainPageIncluded>
<debugInformationGenerated>true</debugInformationGenerated> <debugInformationGenerated>true</debugInformationGenerated>
<sourceMapsGenerated>true</sourceMapsGenerated> <sourceMapsGenerated>true</sourceMapsGenerated>
<sourceFilesCopied>true</sourceFilesCopied>
<targetDirectory>${project.build.directory}/javascript/hello</targetDirectory> <targetDirectory>${project.build.directory}/javascript/hello</targetDirectory>
</configuration> </configuration>
</execution> </execution>