org.pitest.maven.MojoToReportOptionsConverter Maven / Gradle / Ivy
/*
* Copyright 2011 Henry Coles
*
* 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.pitest.maven;
import org.apache.maven.artifact.Artifact;
import org.apache.maven.artifact.DependencyResolutionRequiredException;
import org.apache.maven.model.Build;
import org.apache.maven.model.Plugin;
import org.apache.maven.plugin.logging.Log;
import org.apache.maven.project.MavenProject;
import org.codehaus.plexus.util.xml.Xpp3Dom;
import org.eclipse.aether.artifact.DefaultArtifact;
import org.eclipse.aether.resolution.ArtifactRequest;
import org.eclipse.aether.resolution.ArtifactResolutionException;
import org.eclipse.aether.resolution.ArtifactResult;
import org.pitest.classinfo.ClassName;
import org.pitest.classpath.DirectoryClassPathRoot;
import org.pitest.functional.FCollection;
import org.pitest.mutationtest.config.ExecutionMode;
import org.pitest.mutationtest.config.ReportOptions;
import org.pitest.testapi.TestGroupConfig;
import org.pitest.util.Glob;
import org.pitest.util.Verbosity;
import java.io.File;
import java.nio.file.FileSystems;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.Collections;
import java.util.Enumeration;
import java.util.HashSet;
import java.util.List;
import java.util.Objects;
import java.util.Optional;
import java.util.Properties;
import java.util.Set;
import java.util.function.Function;
import java.util.function.Predicate;
import java.util.stream.Collectors;
import java.util.stream.Stream;
import static org.pitest.functional.Streams.asStream;
public class MojoToReportOptionsConverter {
private final AbstractPitMojo mojo;
private final Predicate dependencyFilter;
private final Log log;
private final SurefireConfigConverter surefireConverter;
public MojoToReportOptionsConverter(final AbstractPitMojo mojo,
SurefireConfigConverter surefireConverter,
Predicate dependencyFilter) {
this.mojo = mojo;
this.dependencyFilter = dependencyFilter;
this.log = mojo.getLog();
this.surefireConverter = surefireConverter;
}
public ReportOptions convert() {
final List classPath = new ArrayList<>();
try {
classPath.addAll(this.mojo.getProject().getTestClasspathElements());
} catch (final DependencyResolutionRequiredException e1) {
this.log.info(e1);
}
addOwnDependenciesToClassPath(classPath);
classPath.addAll(this.mojo.getAdditionalClasspathElements());
autoAddJUnitPlatform(classPath);
removeExcludedDependencies(classPath);
addCrossModuleDirsToClasspath(classPath);
ReportOptions option = parseReportOptions(classPath);
ReportOptions withSureFire = updateFromSurefire(option);
// Null check here is a bad unit testing artifact, should never be null in real life
ReportOptions effective = withSureFire != null ? withSureFire : option;
// argline may contain surefire style properties that require expanding
if (effective.getArgLine() != null) {
log.info("Replacing properties in argLine " + effective.getArgLine());
effective.setArgLine(this.replacePropertyExpressions(effective.getArgLine()));
}
return effective;
}
private void addCrossModuleDirsToClasspath(List classPath) {
// Add the output directories modules we depend on to the start of the classpath.
// If we resolve cross project classes from a jar, the path match
// will fail. This is only an issue when running the pitest goal directly.
if (mojo.isCrossModule()) {
classPath.addAll(0, crossModuleDependencies());
}
}
/**
* The junit 5 plugin needs junit-platform-launcher to run, but this will not be on the classpath
* of the project. We want to use the same version that surefire (and therefore the SUT) uses, not
* the one the plugin was built against.
*
* It is not declared as a normal dependency, instead surefire picks the version to use based on
* other junit jars on the classpath. We're forced to do something similar here.
*
* @param classPath classpath to modify
*/
private void autoAddJUnitPlatform(List classPath) {
List junitDependencies = this.mojo.getProject().getArtifacts().stream()
.filter(a -> a.getGroupId().equals("org.junit.platform"))
.collect(Collectors.toList());
// If the launcher has been manually added to the dependencies, there is nothing to do
if (junitDependencies.stream().anyMatch(a -> a.getArtifactId().equals("junit-platform-launcher"))) {
return;
}
Optional maybeJUnitPlatform = findJUnitArtifact(junitDependencies);
if (!maybeJUnitPlatform.isPresent()) {
this.log.debug("JUnit 5 not on classpath");
return;
}
// Look for platform engine or platform commons on classpath
Artifact toMatch = maybeJUnitPlatform.get();
// Assume that platform launcher has been released with same version number as engine and commons
DefaultArtifact platformLauncher = new DefaultArtifact(toMatch.getGroupId(), "junit-platform-launcher", "jar",
toMatch.getVersion());
try {
ArtifactRequest r = new ArtifactRequest();
r.setArtifact(platformLauncher);
r.setRepositories(this.mojo.getProject().getRemotePluginRepositories());
ArtifactResult resolved = this.mojo.repositorySystem().resolveArtifact(mojo.session().getRepositorySession(), r);
this.log.info("Auto adding " + resolved + " to classpath.");
classPath.add(resolved.getArtifact().getFile().getAbsolutePath());
} catch (ArtifactResolutionException e) {
this.log.error("Could not resolve " + platformLauncher);
throw new RuntimeException(e);
}
}
private static Optional findJUnitArtifact(List junitDependencies) {
Optional maybeEngine = junitDependencies.stream()
.filter(a -> a.getArtifactId().equals("junit-platform-engine"))
.findAny();
if (maybeEngine.isPresent()) {
return maybeEngine;
}
return junitDependencies.stream()
.filter(a -> a.getArtifactId().equals("junit-platform-commons"))
.findAny();
}
private void removeExcludedDependencies(List classPath) {
for (Object artifact : this.mojo.getProject().getArtifacts()) {
final Artifact dependency = (Artifact) artifact;
if (this.mojo.getClasspathDependencyExcludes().contains(
dependency.getGroupId() + ":" + dependency.getArtifactId())) {
classPath.remove(dependency.getFile().getPath());
}
}
}
private ReportOptions parseReportOptions(final List classPath) {
final ReportOptions data = new ReportOptions();
if (this.mojo.getProject().getBuild() != null) {
List codePaths = new ArrayList<>();
codePaths.add(this.mojo.getProject().getBuild()
.getOutputDirectory());
if (mojo.isCrossModule()) {
codePaths.addAll(crossModuleDependencies());
}
this.log.info("Mutating from "
+ String.join(",", codePaths));
data.setCodePaths(Collections.singleton(this.mojo.getProject().getBuild()
.getOutputDirectory()));
data.setCodePaths(codePaths);
}
data.setUseClasspathJar(this.mojo.isUseClasspathJar());
data.setClassPathElements(classPath);
data.setFailWhenNoMutations(shouldFailWhenNoMutations());
data.setTargetClasses(determineTargetClasses());
data.setTargetTests(determineTargetTests());
data.setExcludedMethods(this.mojo
.getExcludedMethods());
data.setExcludedClasses(this.mojo.getExcludedClasses());
data.setExcludedTestClasses(globStringsToPredicates(this.mojo
.getExcludedTestClasses()));
data.setNumberOfThreads(this.mojo.getThreads());
data.setExcludedRunners(this.mojo.getExcludedRunners());
data.setReportDir(this.mojo.getReportsDirectory().getAbsolutePath());
configureVerbosity(data);
if (this.mojo.getJvmArgs() != null) {
data.addChildJVMArgs(this.mojo.getJvmArgs());
}
if (this.mojo.getArgLine() != null) {
data.setArgLine(this.mojo.getArgLine());
}
data.setMutators(determineMutators());
data.setFeatures(determineFeatures());
data.setTimeoutConstant(this.mojo.getTimeoutConstant());
data.setTimeoutFactor(this.mojo.getTimeoutFactor());
if (hasValue(this.mojo.getAvoidCallsTo())) {
data.setLoggingClasses(this.mojo.getAvoidCallsTo());
}
final List sourceRoots = determineSourceRoots();
data.setSourceDirs(stringsToPaths(sourceRoots));
data.addOutputFormats(determineOutputFormats());
setTestGroups(data);
data.setFullMutationMatrix(this.mojo.isFullMutationMatrix());
data.setMutationUnitSize(this.mojo.getMutationUnitSize());
data.setShouldCreateTimestampedReports(this.mojo.isTimestampedReports());
data.setDetectInlinedCode(this.mojo.isDetectInlinedCode());
determineHistory(data);
data.setExportLineCoverage(this.mojo.isExportLineCoverage());
data.setMutationEngine(this.mojo.getMutationEngine());
data.setJavaExecutable(this.mojo.getJavaExecutable());
data.setFreeFormProperties(createPluginProperties());
data.setIncludedTestMethods(this.mojo.getIncludedTestMethods());
data.setSkipFailingTests(this.mojo.skipFailingTests());
data.setInputEncoding(this.mojo.getSourceEncoding());
data.setOutputEncoding(this.mojo.getOutputEncoding());
if (this.mojo.getProjectBase() != null) {
data.setProjectBase(FileSystems.getDefault().getPath(this.mojo.getProjectBase()));
}
if (this.mojo.isDryRun()) {
data.setExecutionMode(ExecutionMode.DRY_RUN);
}
checkForObsoleteOptions(this.mojo);
return data;
}
private List determineSourceRoots() {
final List sourceRoots = new ArrayList<>();
sourceRoots.addAll(this.mojo.getProject().getCompileSourceRoots());
sourceRoots.addAll(this.mojo.getProject().getTestCompileSourceRoots());
if (mojo.isCrossModule()) {
List otherRoots = dependedOnProjects().stream()
.flatMap(p -> p.getCompileSourceRoots().stream())
.collect(Collectors.toList());
sourceRoots.addAll(otherRoots);
}
return sourceRoots;
}
private Collection crossModuleDependencies() {
return dependedOnProjects().stream()
.map(MavenProject::getBuild)
.map(Build::getOutputDirectory)
.filter(Objects::nonNull)
.collect(Collectors.toList());
}
private List dependedOnProjects() {
// strip version from dependencies
Set inScope = this.mojo.getProject().getDependencies().stream()
.map(p -> p.getGroupId() + ":" + p.getArtifactId())
.collect(Collectors.toSet());
return this.mojo.allProjects().stream()
.filter(p -> inScope.contains(p.getGroupId() + ":" + p.getArtifactId()))
.collect(Collectors.toList());
}
private void configureVerbosity(ReportOptions data) {
if (this.mojo.isVerbose()) {
data.setVerbosity(Verbosity.VERBOSE);
} else {
Verbosity v = Verbosity.fromString(mojo.getVerbosity());
data.setVerbosity(v);
}
}
private void checkForObsoleteOptions(AbstractPitMojo mojo) {
if (mojo.getMaxMutationsPerClass() > 0) {
throw new IllegalArgumentException("The max mutations per class argument is no longer supported, "
+ "use features=+CLASSLIMIT(limit[" + mojo.getMaxMutationsPerClass() + "]) instead");
}
}
private void determineHistory(final ReportOptions data) {
// set explicit history files if configured
data.setHistoryInputLocation(this.mojo.getHistoryInputFile());
data.setHistoryOutputLocation(this.mojo.getHistoryOutputFile());
// If withHistory option set, overwrite config with files in temp dir.
// This allows a user to configure files for use on ci, but still easily use temp files
// for local running
if (this.mojo.useHistory()) {
useHistoryFileInTempDir(data);
}
if (data.getHistoryInputLocation() != null) {
log.info("Will read history at " + data.getHistoryInputLocation());
}
if (data.getHistoryOutputLocation() != null) {
log.info("Will write history at " + data.getHistoryOutputLocation());
}
}
private void useHistoryFileInTempDir(final ReportOptions data) {
String tempDir = System.getProperty("java.io.tmpdir");
MavenProject project = this.mojo.getProject();
String name = project.getGroupId() + "."
+ project.getArtifactId() + "."
+ project.getVersion() + "_pitest_history.bin";
File historyFile = new File(tempDir, name);
if (mojo.getHistoryInputFile() != null || mojo.getHistoryOutputFile() != null) {
log.info("Using withHistory option. This overrides the explicitly set history file paths.");
}
data.setHistoryInputLocation(historyFile);
data.setHistoryOutputLocation(historyFile);
}
private ReportOptions updateFromSurefire(ReportOptions option) {
Collection plugins = lookupPlugin("org.apache.maven.plugins:maven-surefire-plugin");
if (!this.mojo.isParseSurefireConfig()) {
return option;
} else if (plugins.isEmpty()) {
this.log.warn("Could not find surefire configuration in pom");
return option;
}
Plugin surefire = plugins.iterator().next();
if (surefire != null) {
return this.surefireConverter.update(option,
(Xpp3Dom) surefire.getConfiguration());
} else {
return option;
}
}
private Collection lookupPlugin(String key) {
List plugins = this.mojo.getProject().getBuildPlugins();
return FCollection.filter(plugins, hasKey(key));
}
private static Predicate hasKey(final String key) {
return a -> a.getKey().equals(key);
}
private boolean shouldFailWhenNoMutations() {
return this.mojo.isFailWhenNoMutations();
}
private void setTestGroups(final ReportOptions data) {
final TestGroupConfig conf = new TestGroupConfig(
this.mojo.getExcludedGroups(), this.mojo.getIncludedGroups());
data.setGroupConfig(conf);
}
private void addOwnDependenciesToClassPath(final List classPath) {
for (final Artifact dependency : filteredDependencies()) {
this.log.info("Adding " + dependency.getGroupId() + ":"
+ dependency.getArtifactId() + " to SUT classpath");
classPath.add(dependency.getFile().getAbsolutePath());
}
// If the user as explicitly added junit platform classes to the pitest classpath, trust that they
// know what they're doing and add them in
this.mojo.getPluginArtifactMap().values().stream().filter(a -> a.getGroupId().equals("org.junit.platform"))
.forEach(dependency -> classPath.add(dependency.getFile().getAbsolutePath()));
}
private Collection> globStringsToPredicates(
final List excludedMethods) {
return asStream(excludedMethods)
.map(Glob.toGlobPredicate())
.collect(Collectors.toList());
}
private Collection> determineTargetTests() {
return useConfiguredTargetTestsOrFindOccupiedPackages(this.mojo.getTargetTests()).stream()
.map(Glob.toGlobPredicate())
.collect(Collectors.toList());
}
private Collection useConfiguredTargetTestsOrFindOccupiedPackages(
final Collection filters) {
if (!hasValue(filters)) {
this.mojo.getLog().info("Defaulting target tests to match packages in test build directory");
return findOccupiedTestPackages();
} else {
return filters;
}
}
private Collection findOccupiedTestPackages() {
// use only the tests within current project, even if in
// cross module mode
String outputDirName = this.mojo.getProject().getBuild()
.getTestOutputDirectory();
if (outputDirName != null) {
File outputDir = new File(outputDirName);
return findOccupiedPackagesIn(outputDir);
} else {
return Collections.emptyList();
}
}
private Collection filteredDependencies() {
return FCollection.filter(this.mojo.getPluginArtifactMap().values(),
this.dependencyFilter);
}
private Collection determineMutators() {
if (this.mojo.getMutators() != null) {
return this.mojo.getMutators();
} else {
return Collections.emptyList();
}
}
private Collection determineFeatures() {
if (this.mojo.getFeatures() != null) {
return this.mojo.getFeatures();
} else {
return Collections.emptyList();
}
}
private Collection determineTargetClasses() {
return useConfiguredTargetClassesOrFindOccupiedPackages(this.mojo.getTargetClasses());
}
private Collection useConfiguredTargetClassesOrFindOccupiedPackages(
final Collection filters) {
if (!hasValue(filters)) {
this.mojo.getLog().info("Defaulting target classes to match packages in build directory");
return findOccupiedPackages();
} else {
return filters;
}
}
private Collection findOccupiedPackages() {
return Stream.concat(Stream.of(mojo.getProject()), dependedOnProjects().stream())
.distinct()
.map(p -> new File(p.getBuild().getOutputDirectory()))
.flatMap(f -> findOccupiedPackagesIn(f).stream())
.distinct()
.collect(Collectors.toList());
}
public static Collection findOccupiedPackagesIn(File dir) {
if (dir.exists()) {
DirectoryClassPathRoot root = new DirectoryClassPathRoot(dir);
Set occupiedPackages = new HashSet<>();
FCollection.mapTo(root.classNames(), classToPackageGlob(),
occupiedPackages);
return occupiedPackages;
}
return Collections.emptyList();
}
private static Function classToPackageGlob() {
return a -> ClassName.fromString(a).getPackage().asJavaName() + ".*";
}
private Collection stringsToPaths(final List sourceRoots) {
return sourceRoots.stream()
.map(Paths::get)
.collect(Collectors.toList());
}
private Collection determineOutputFormats() {
if (hasValue(this.mojo.getOutputFormats())) {
return this.mojo.getOutputFormats();
} else {
return Arrays.asList("HTML");
}
}
private boolean hasValue(final Collection> collection) {
return (collection != null) && !collection.isEmpty();
}
private Properties createPluginProperties() {
Properties p = new Properties();
if (this.mojo.getPluginProperties() != null) {
p.putAll(this.mojo.getPluginProperties());
}
return p;
}
/**
* This method taken from surefire under Apache 2 licence.
*
* Replaces expressions @{property-name}
with the corresponding properties
* from the model. This allows late evaluation of property values when the plugin is executed (as compared
* to evaluation when the pom is parsed as is done with ${property-name}
expressions).
*
* This allows other plugins to modify or set properties with the changes getting picked up by surefire.
*/
private String replacePropertyExpressions(String argLine) {
for (Enumeration> e = mojo.getProject().getProperties().propertyNames(); e.hasMoreElements();) {
String key = e.nextElement().toString();
// Replace surefire late evaluation syntax properties
argLine = replaceFieldForSymbol('@', key, argLine);
// Normally properties will already have been expanded by maven, but this is
// bypassed for argLines pulled from surefire, se we must handle them here
argLine = replaceFieldForSymbol('$', key, argLine);
}
return argLine;
}
private String replaceFieldForSymbol(char symbol, String key, String argLine) {
String field = symbol + "{" + key + "}";
if (argLine.contains(field)) {
return argLine.replace(field, mojo.getProject().getProperties().getProperty(key, ""));
}
return argLine;
}
}