blob: fa09f1211b90e8ca0d8732894f08182b88bc1e0f [file] [log] [blame]
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You 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.apache.jackrabbit.filevault.maven.packaging;
import java.io.IOException;
import java.util.Collection;
import java.util.Comparator;
import java.util.HashMap;
import java.util.LinkedList;
import java.util.Locale;
import java.util.Map;
import java.util.stream.Collectors;
import org.apache.commons.lang3.StringUtils;
import org.apache.jackrabbit.filevault.maven.packaging.validator.impl.context.DependencyResolver;
import org.apache.jackrabbit.vault.fs.api.PathFilterSet;
import org.apache.jackrabbit.vault.packaging.Dependency;
import org.apache.jackrabbit.vault.packaging.PackageId;
import org.apache.jackrabbit.vault.packaging.PackageInfo;
import org.apache.jackrabbit.vault.util.Constants;
import org.apache.jackrabbit.vault.validation.ValidationExecutorFactory;
import org.apache.jackrabbit.vault.validation.spi.ValidationMessageSeverity;
import org.apache.jackrabbit.vault.validation.spi.impl.AdvancedFilterValidatorFactory;
import org.apache.jackrabbit.vault.validation.spi.impl.DependencyValidatorFactory;
import org.apache.maven.artifact.Artifact;
import org.apache.maven.artifact.DefaultArtifact;
import org.apache.maven.artifact.repository.DefaultRepositoryRequest;
import org.apache.maven.artifact.resolver.ResolutionErrorHandler;
import org.apache.maven.execution.MavenSession;
import org.apache.maven.plugin.AbstractMojo;
import org.apache.maven.plugin.MojoExecution;
import org.apache.maven.plugin.MojoExecutionException;
import org.apache.maven.plugin.MojoFailureException;
import org.apache.maven.plugin.logging.Log;
import org.apache.maven.plugins.annotations.Component;
import org.apache.maven.plugins.annotations.Parameter;
import org.apache.maven.project.MavenProject;
import org.apache.maven.repository.RepositorySystem;
import org.sonatype.plexus.build.incremental.BuildContext;
/**
* Common ancestor for all validation related mojos
*/
public abstract class AbstractValidateMojo extends AbstractMojo {
public static final String IGNORE_GAV = "ignore";
@Parameter(property = "vault.skipValidation", defaultValue = "false", required = true)
boolean skipValidation;
/** All validator settings in a map. The keys are the validator ids (optionally suffixed by {@code :<package group>:<package name>} to be restricted to certain packages).
* You can use {@code *} as wildcard value for {@code package group}.
* Alternatively you can use the suffix {@code :subpackages} to influence the settings for all sub packages only!
* The values are a complex object of type ValdidatorSettings.
* An example configuration looks like
* <pre>
* &lt;jackrabbit-filter&gt;
* &lt;options&gt;
* &lt;severityForUncoveredAncestorNodes&gt;error&lt;/severityForUncoveredAncestorNodes&gt;
* &lt;/options&gt;
* &lt;/jackrabbit-filter&gt;
* </pre>
*/
@Parameter
private Map<String, ValidatorSettings> validatorsSettings;
/** Controls if errors during dependency validation should fail the build.
*
* @deprecated Use {@link validatorsSettings} with the following values
* instead
* <pre>
* &lt;jackrabbit-dependencies&gt;
* &lt;defaultSeverity&gt;debug&lt;/defaultSeverity&gt;
* &lt;/jackrabbit-dependencies&gt;
* </pre>
*
*/
@Parameter(property = "vault.failOnDependencyErrors", defaultValue = "true", required = true)
@Deprecated
private boolean failOnDependencyErrors;
/** The Maven project (might be {@code null}) */
@Parameter(property = "project", readonly = true, required = false)
protected MavenProject project;
@Parameter(defaultValue = "${mojoExecution}", readonly = true, required = true)
protected MojoExecution mojoExecution;
@Parameter(defaultValue = "${session}", readonly = true, required = false)
protected MavenSession session;
/** If set to {@code true} will lead to all validation errors or warnings failing the build, otherwise only validation errors lead to a
* build failure */
@Parameter(property = "vault.failOnValidationWarning", defaultValue = "false")
protected boolean failOnValidationWarnings;
/** Defines the list of dependencies A dependency is declared as a {@code <dependency>} element of a list style {@code <dependencies>}
* element:
*
* <pre>
* &lt;dependency&gt;
* &lt;group&gt;theGroup&lt;/group&gt;
* &lt;name&gt;theName&lt;/name&gt;
* &lt;version&gt;1.5&lt;/version&gt;
* &lt;/dependency&gt;
* </pre>
* <p>
* The dependency can also reference a maven project dependency, this is preferred as it yields to more robust builds.
*
* <pre>
* &lt;dependency&gt;
* &lt;groupId&gt;theGroup&lt;/groupId&gt;
* &lt;artifactId&gt;theName&lt;/artifactId&gt;
* &lt;/dependency&gt;
* </pre>
* <p>
* The {@code versionRange} may be indicated as a single version, in which case the version range has no upper bound and defines the
* minimal version accepted. Otherwise, the version range defines a lower and upper bound of accepted versions, where the bounds are
* either included using parentheses {@code ()} or excluded using brackets {@code []} */
@Parameter(property = "vault.dependencies")
protected Collection<MavenBasedPackageDependency> dependencies = new LinkedList<>();
/** Defines the packages that define the repository structure. For the format description look at {@link #dependencies}.
* <p>
* The repository-init feature of sling-start can define initial content that will be available in the repository before the first
* package is installed. Packages that depend on those nodes have no way to reference any dependency package that provides these nodes.
* A "real" package that would creates those nodes cannot be installed in the repository, because it would void the repository init
* structure. On the other hand would filevault complain, if the package was listed as dependency but not installed in the repository.
* So therefore this repository-structure packages serve as indicator packages that helps satisfy the structural dependencies, but are
* not added as real dependencies to the package. */
@Parameter(property = "vault.repository.structure.packages")
protected Collection<MavenBasedPackageDependency> repositoryStructurePackages = new LinkedList<>();
/**
* Mapping of package dependencies given via group and name to Maven identifiers for enhanced validation.
* Each entry must have the format {@code <group>:<name>=<groupId>:<artifactId>}.
* To disable lookup (e.g. because referenced artifact is not available in a Maven repository) use {@code <group>:<name>=ignore}.
* This will also prevent the WARNING which would be otherwise be emitted.
*/
@Parameter(property = "vault.package.dependency.to.maven.ga")
protected Collection<String> mapPackageDependencyToMavenGa;
@Component
protected RepositorySystem repositorySystem;
@Component
protected ResolutionErrorHandler resolutionErrorHandler;
@Component
protected BuildContext buildContext;
protected final ValidationExecutorFactory validationExecutorFactory;
protected final ValidationHelper validationHelper;
protected DependencyResolver resolver;
/**
* Artificial Maven artifact which indicates that it should not be considered for further lookup!
*/
public static final Artifact IGNORE_ARTIFACT = new DefaultArtifact("ignore", "ignore", "1.0", "", "", "", null);
public AbstractValidateMojo() {
super();
this.validationExecutorFactory = new ValidationExecutorFactory(this.getClass().getClassLoader());
this.validationHelper = new ValidationHelper();
}
static Map<Dependency, Artifact> resolveMap(Collection<String> mapPackageDependencyToMavenGa) {
// resolve mapping map
return mapPackageDependencyToMavenGa.stream()
.map(s -> s.split("=", 2))
.peek((p) -> { if(p.length != 2) { throw new IllegalArgumentException("Could not parse value"); } })
// cannot use null values due to https://bugs.openjdk.java.net/browse/JDK-8148463 therefore rely on artificial IGNORE_ARTIFACT
.collect(Collectors.toMap(a -> Dependency.fromString(a[0]), a -> { if (a[1].equalsIgnoreCase(IGNORE_GAV)) { return IGNORE_ARTIFACT; } String[] mavenGA = a[1].split(":", 2); if(mavenGA.length != 2) { throw new IllegalArgumentException("Could not parse Maven group Id and artifact Id (must be separated by ':')"); } return new DefaultArtifact(mavenGA[0], mavenGA[1], "", "", "", "", null);} ));
}
/**
*
* @return {@code true} to skip execution of the mojo. Default is {@code false}.
*/
protected boolean shouldSkip() {
return false;
}
@Override
public void execute() throws MojoExecutionException, MojoFailureException {
if (shouldSkip()) {
return;
}
if (skipValidation) {
getLog().info("Skipping validation");
return;
}
translateLegacyParametersToValidatorParameters();
final Collection<PackageInfo> resolvedDependencies = new LinkedList<>();
if (project != null) {
validationHelper.clearPreviousValidationMessages(buildContext, project.getBasedir());
}
// repository structure only defines valid roots
// https://github.com/apache/jackrabbit-filevault-package-maven-plugin/blob/02a853e64d985f075fe88d19101d7c66d741767f/src/main/java/org/apache/jackrabbit/filevault/maven/packaging/impl/DependencyValidator.java#L51
try {
Collection<String> validRoots = new LinkedList<>();
for (PackageInfo packageInfo : getPackageInfoFromMavenBasedDependencies(repositoryStructurePackages)) {
for (PathFilterSet set : packageInfo.getFilter().getFilterSets()) {
validRoots.add(set.getRoot());
}
}
if (!validRoots.isEmpty()) {
ValidatorSettings settings = null;
if (validatorsSettings != null) {
settings = validatorsSettings.get(AdvancedFilterValidatorFactory.ID);
} else {
validatorsSettings = new HashMap<>();
}
if (settings == null) {
settings = new ValidatorSettings();
settings.addOption(AdvancedFilterValidatorFactory.OPTION_VALID_ROOTS, StringUtils.join(validRoots, ","));
validatorsSettings.put(AdvancedFilterValidatorFactory.ID, settings);
} else {
String oldValidRoots = settings.getOptions().get(AdvancedFilterValidatorFactory.OPTION_VALID_ROOTS);
settings.addOption(AdvancedFilterValidatorFactory.OPTION_VALID_ROOTS, oldValidRoots + "," + StringUtils.join(validRoots, ","));
}
}
} catch (IOException e) {
throw new MojoExecutionException("Could not get meta information for repositoryStructurePackages '"
+ StringUtils.join(repositoryStructurePackages, ",") + "': " + e.getMessage(), e);
}
try {
resolvedDependencies.addAll(getPackageInfoFromMavenBasedDependencies(dependencies));
} catch (IOException e) {
throw new MojoExecutionException(
"Could not get meta information for dependencies '" + StringUtils.join(dependencies, ",") + "': " + e.getMessage(),
e);
}
// resolve mapping map
resolver = new DependencyResolver(DefaultRepositoryRequest.getRepositoryRequest(session, project), repositorySystem,
resolutionErrorHandler, resolveMap(mapPackageDependencyToMavenGa), resolvedDependencies);
doExecute();
}
private Collection<PackageInfo> getPackageInfoFromMavenBasedDependencies(Collection<MavenBasedPackageDependency> dependencies) throws IOException {
Collection<PackageInfo> packageInfos = new LinkedList<>();
// try to resolve from project artifacts (in case a project is given)
MavenBasedPackageDependency.resolve(project, getLog(), dependencies);
for (MavenBasedPackageDependency dependency : dependencies) {
// check if resolution was successful
if (dependency.getInfo() != null) {
packageInfos.add(dependency.getInfo());
}
}
return packageInfos;
}
private void translateLegacyParametersToValidatorParameters() throws MojoExecutionException {
if (!failOnDependencyErrors) {
getLog().warn("Deprecated parameter 'failOnDependencyErrors' used.");
if (validatorsSettings != null) {
if (validatorsSettings.containsKey(DependencyValidatorFactory.ID)) {
throw new MojoExecutionException("Can not set parameters 'failOnDependencyErrors' and 'validationSettings' for '"
+ DependencyValidatorFactory.ID + "' at the same time");
}
} else {
validatorsSettings = new HashMap<>();
}
// do no fully disable but emit violations with level DEBUG
ValidatorSettings dependencyValidatorSettings = new ValidatorSettings(ValidationMessageSeverity.DEBUG);
validatorsSettings.put(DependencyValidatorFactory.ID, dependencyValidatorSettings);
ValidatorSettings filterValidatorSettings = validatorsSettings.containsKey(AdvancedFilterValidatorFactory.ID) ? validatorsSettings.get(AdvancedFilterValidatorFactory.ID) : new ValidatorSettings();
if (filterValidatorSettings.getOptions().containsKey(AdvancedFilterValidatorFactory.OPTION_SEVERITY_FOR_UNCOVERED_ANCESTOR_NODES)) {
throw new MojoExecutionException("Can not set parameters 'failOnDependencyErrors' and 'validationSettings' for '"
+ DependencyValidatorFactory.ID + "' with option'" +AdvancedFilterValidatorFactory.OPTION_SEVERITY_FOR_UNCOVERED_ANCESTOR_NODES + "' at the same time");
}
filterValidatorSettings.addOption(AdvancedFilterValidatorFactory.OPTION_SEVERITY_FOR_UNCOVERED_ANCESTOR_NODES, ValidationMessageSeverity.DEBUG.toString().toLowerCase(Locale.ROOT));
}
}
protected void disableChecksOnlyWorkingForPackages() throws MojoExecutionException {
final ValidatorSettings filterValidatorSettings;
if (validatorsSettings == null) {
validatorsSettings = new HashMap<>();
}
if (validatorsSettings.containsKey(AdvancedFilterValidatorFactory.ID)) {
getLog().warn("Overwriting settings for validator " + AdvancedFilterValidatorFactory.ID + " as some checks do not work reliably for this mojo!");
filterValidatorSettings = validatorsSettings.get(AdvancedFilterValidatorFactory.ID);
} else {
filterValidatorSettings = new ValidatorSettings();
}
// orphaned filter rules cannot be realiably detected, as the package is not yet build
filterValidatorSettings.addOption(AdvancedFilterValidatorFactory.OPTION_SEVERITY_FOR_ORPHANED_FILTER_RULES, "debug");
}
public abstract void doExecute() throws MojoExecutionException, MojoFailureException;
protected Map<String, ValidatorSettings> getValidatorSettingsForPackage(PackageId packageId, boolean isSubPackage) {
return getValidatorSettingsForPackage(getLog(), validatorsSettings, packageId, isSubPackage);
}
static Map<String, ValidatorSettings> getValidatorSettingsForPackage(Log log, Map<String, ValidatorSettings> validatorsSettings, PackageId packageId, boolean isSubPackage) {
Map<String, ValidatorSettings> validatorSettingsById = new HashMap<>();
if (validatorsSettings == null) {
return validatorSettingsById;
}
for (Map.Entry<String, ValidatorSettings> validatorSettingByIdAndPackage : validatorsSettings.entrySet()) {
// does this setting belong to this package?
boolean shouldAdd = false;
String[] parts = validatorSettingByIdAndPackage.getKey().split(":", 3);
final String validatorId = parts[0];
if (parts.length == 2) {
if (parts[1].equals("subpackage")) {
shouldAdd = isSubPackage;
} else {
log.warn("Invalid validatorSettings key '" + validatorSettingByIdAndPackage.getKey() +"'" );
continue;
}
}
// does it contain a package id filter?
else if (parts.length == 3) {
String group = parts[1];
String name = parts[2];
if (group != "*") {
if (!group.equals(packageId.getGroup())) {
log.debug("Not applying validator settings with id '" + validatorSettingByIdAndPackage.getKey() +"' as it does not match the package " + packageId);
continue;
}
}
if (!name.equals(packageId.getName())) {
log.debug("Not applying validator settings with id '" + validatorSettingByIdAndPackage.getKey() +"' as it does not match the package " + packageId);
continue;
}
shouldAdd = true;
} else {
shouldAdd = true;
}
if (shouldAdd) {
validatorSettingsById.put(validatorId, validatorSettingByIdAndPackage.getValue());
}
}
return validatorSettingsById;
}
/**
* Comparator on file names which makes sure that the {@code .content.xml} files come first.
*/
static final class DotContentXmlFirstComparator implements Comparator<String> {
@Override
public int compare(String s1, String s2) {
if (Constants.DOT_CONTENT_XML.equals(s1)) {
return -1;
} else if (Constants.DOT_CONTENT_XML.equals(s2)) {
return 1;
}
return 0;
}
}
}