org.apache.maven.plugin.resources.remote.AbstractProcessRemoteResourcesMojo Maven / Gradle / Ivy
Show all versions of maven-remote-resources-plugin Show documentation
/*
* 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.maven.plugin.resources.remote;
import java.io.File;
import java.io.FileOutputStream;
import java.io.FileReader;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.io.OutputStreamWriter;
import java.io.Reader;
import java.io.StringReader;
import java.io.Writer;
import java.net.MalformedURLException;
import java.net.URL;
import java.nio.charset.Charset;
import java.nio.file.Files;
import java.time.Instant;
import java.time.ZoneId;
import java.time.format.DateTimeFormatter;
import java.util.AbstractMap;
import java.util.ArrayList;
import java.util.Collections;
import java.util.Comparator;
import java.util.Enumeration;
import java.util.HashMap;
import java.util.LinkedHashSet;
import java.util.List;
import java.util.Map;
import java.util.Properties;
import java.util.Set;
import java.util.TreeMap;
import org.apache.maven.RepositoryUtils;
import org.apache.maven.archiver.MavenArchiver;
import org.apache.maven.artifact.Artifact;
import org.apache.maven.artifact.handler.manager.ArtifactHandlerManager;
import org.apache.maven.execution.MavenSession;
import org.apache.maven.model.Model;
import org.apache.maven.model.Organization;
import org.apache.maven.model.Resource;
import org.apache.maven.model.building.ModelBuildingRequest;
import org.apache.maven.model.io.xpp3.MavenXpp3Reader;
import org.apache.maven.plugin.AbstractMojo;
import org.apache.maven.plugin.MojoExecutionException;
import org.apache.maven.plugin.resources.remote.io.xpp3.RemoteResourcesBundleXpp3Reader;
import org.apache.maven.plugin.resources.remote.io.xpp3.SupplementalDataModelXpp3Reader;
import org.apache.maven.plugins.annotations.Parameter;
import org.apache.maven.project.DefaultProjectBuildingRequest;
import org.apache.maven.project.MavenProject;
import org.apache.maven.project.ProjectBuilder;
import org.apache.maven.project.ProjectBuildingException;
import org.apache.maven.project.ProjectBuildingRequest;
import org.apache.maven.project.ProjectBuildingResult;
import org.apache.maven.shared.artifact.filter.collection.ArtifactFilterException;
import org.apache.maven.shared.artifact.filter.collection.ArtifactIdFilter;
import org.apache.maven.shared.artifact.filter.collection.FilterArtifacts;
import org.apache.maven.shared.artifact.filter.collection.GroupIdFilter;
import org.apache.maven.shared.artifact.filter.collection.ProjectTransitivityFilter;
import org.apache.maven.shared.artifact.filter.collection.ScopeFilter;
import org.apache.maven.shared.filtering.FilteringUtils;
import org.apache.maven.shared.filtering.MavenFileFilter;
import org.apache.maven.shared.filtering.MavenFileFilterRequest;
import org.apache.maven.shared.filtering.MavenFilteringException;
import org.apache.velocity.VelocityContext;
import org.apache.velocity.app.Velocity;
import org.apache.velocity.app.VelocityEngine;
import org.apache.velocity.exception.MethodInvocationException;
import org.apache.velocity.exception.ParseErrorException;
import org.apache.velocity.exception.ResourceNotFoundException;
import org.apache.velocity.exception.VelocityException;
import org.apache.velocity.runtime.resource.loader.ClasspathResourceLoader;
import org.codehaus.plexus.resource.ResourceManager;
import org.codehaus.plexus.resource.loader.FileResourceLoader;
import org.codehaus.plexus.util.FileUtils;
import org.codehaus.plexus.util.IOUtil;
import org.codehaus.plexus.util.StringUtils;
import org.codehaus.plexus.util.io.CachingOutputStream;
import org.codehaus.plexus.util.xml.Xpp3Dom;
import org.codehaus.plexus.util.xml.pull.XmlPullParserException;
import org.eclipse.aether.RepositorySystem;
import org.eclipse.aether.artifact.ArtifactType;
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.eclipse.aether.util.artifact.JavaScopes;
/**
*
* Pull down resourceBundles containing remote resources and process the resources contained inside. When that is done,
* the resources are injected into the current (in-memory) Maven project, making them available to the process-resources
* phase.
*
*
* Resources that end in ".vm" are treated as Velocity templates. For those, the ".vm" is stripped off for the final
* artifact name and it's fed through Velocity to have properties expanded, conditions processed, etc...
*
* Resources that don't end in ".vm" are copied "as is".
*
* This is a support abstract class, with two non-aggregating and aggregating implementations.
*
*/
public abstract class AbstractProcessRemoteResourcesMojo extends AbstractMojo {
private static final String TEMPLATE_SUFFIX = ".vm";
/**
*
* In cases where a local resource overrides one from a remote resource bundle, that resource should be filtered if
* the resource set specifies it. In those cases, this parameter defines the list of delimiters for filterable
* expressions. These delimiters are specified in the form 'beginToken*endToken'. If no '*' is given, the delimiter
* is assumed to be the same for start and end.
*
*
* So, the default filtering delimiters might be specified as:
*
*
*
* <delimiters>
* <delimiter>${*}</delimiter>
* <delimiter>@</delimiter>
* </delimiters>
*
* Since the '@' delimiter is the same on both ends, we don't need to specify '@*@' (though we can).
*
* @since 1.1
*/
@Parameter
protected List filterDelimiters;
/**
* @since 1.1
*/
@Parameter(defaultValue = "true")
protected boolean useDefaultFilterDelimiters;
/**
* The character encoding scheme to be applied when filtering resources.
*/
@Parameter(property = "encoding", defaultValue = "${project.build.sourceEncoding}")
protected String encoding;
/**
* The directory where processed resources will be placed for packaging.
*/
@Parameter(property = "outputDirectory", defaultValue = "${project.build.directory}/maven-shared-archive-resources")
private File outputDirectory;
/**
* The directory containing extra information appended to the generated resources.
*/
@Parameter(defaultValue = "${basedir}/src/main/appended-resources")
private File appendedResourcesDirectory;
/**
* Supplemental model data. Useful when processing
* artifacts with incomplete POM metadata.
*
* By default, this Mojo looks for supplemental model data in the file
* "${appendedResourcesDirectory}/supplemental-models.xml
".
*
* @since 1.0-alpha-5
*/
@Parameter
private String[] supplementalModels;
/**
* List of artifacts that are added to the search path when looking
* for supplementalModels, expressed with groupId:artifactId:version[:type[:classifier]]
format.
*
* @since 1.1
*/
@Parameter
private List supplementalModelArtifacts;
/**
* The resource bundles that will be retrieved and processed,
* expressed with groupId:artifactId:version[:type[:classifier]]
format.
*/
@Parameter(property = "resourceBundles", required = true)
private List resourceBundles;
/**
* Skip remote-resource processing
*
* @since 1.0-alpha-5
*/
@Parameter(property = "remoteresources.skip", defaultValue = "false")
private boolean skip;
/**
* Attaches the resources to the main build of the project as a resource directory.
*
* @since 1.5
*/
@Parameter(defaultValue = "true", property = "attachToMain")
private boolean attachToMain;
/**
* Attaches the resources to the test build of the project as a resource directory.
*
* @since 1.5
*/
@Parameter(defaultValue = "true", property = "attachToTest")
private boolean attachToTest;
/**
* Additional properties to be passed to Velocity.
* Several properties are automatically added:
* project
- the current MavenProject
* projects
- the list of dependency projects
* projectsSortedByOrganization
- the list of dependency projects sorted by organization
* projectTimespan
- the timespan of the current project (requires inceptionYear in pom)
* locator
- the ResourceManager that can be used to retrieve additional resources
*
* See the
* javadoc for MavenProject for information about the properties on the MavenProject.
*/
@Parameter
protected Map properties = new HashMap<>();
/**
* Whether to include properties defined in the project when filtering resources.
*
* @deprecated as Maven Project is available in Velocity context we can simply use
* $project.properties.propertyName
*
* @since 1.2
*/
@Deprecated
@Parameter(defaultValue = "false")
protected boolean includeProjectProperties = false;
/**
* When the result of velocity transformation fits in memory, it is compared with the actual contents on disk
* to eliminate unnecessary destination file overwrite. This improves build times since further build steps
* typically rely on the modification date.
*
* @deprecated not used anymore
* @since 1.6
*/
@Deprecated
@Parameter(defaultValue = "5242880")
protected int velocityFilterInMemoryThreshold = 5 * 1024 * 1024;
/**
* The Maven session.
*/
@Parameter(defaultValue = "${session}", readonly = true, required = true)
protected MavenSession mavenSession;
/**
* The current project.
*/
@Parameter(defaultValue = "${project}", readonly = true, required = true)
protected MavenProject project;
/**
* Scope to include. An Empty string indicates all scopes (default is "runtime").
*
* @since 1.0
*/
@Parameter(property = "includeScope", defaultValue = "runtime")
protected String includeScope;
/**
* Scope to exclude. An Empty string indicates no scopes (default).
*
* @since 1.0
*/
@Parameter(property = "excludeScope", defaultValue = "")
protected String excludeScope;
/**
* When resolving project dependencies, specify the scopes to include.
* The default is the same as "includeScope" if there are no exclude scopes set.
* Otherwise, it defaults to "test" to grab all the dependencies so the
* exclude filters can filter out what is not needed.
*
* @since 1.5
*/
@Parameter
protected String[] resolveScopes;
/**
* Comma separated list of Artifact names to exclude.
*
* @since 1.0
*/
@Parameter(property = "excludeArtifactIds", defaultValue = "")
protected String excludeArtifactIds;
/**
* Comma separated list of Artifact names to include.
*
* @since 1.0
*/
@Parameter(property = "includeArtifactIds", defaultValue = "")
protected String includeArtifactIds;
/**
* Comma separated list of GroupId Names to exclude.
*
* @since 1.0
*/
@Parameter(property = "excludeGroupIds", defaultValue = "")
protected String excludeGroupIds;
/**
* Comma separated list of GroupIds to include.
*
* @since 1.0
*/
@Parameter(property = "includeGroupIds", defaultValue = "")
protected String includeGroupIds;
/**
* If we should exclude transitive dependencies
*
* @since 1.0
*/
@Parameter(property = "excludeTransitive", defaultValue = "false")
protected boolean excludeTransitive;
/**
* Timestamp for reproducible output archive entries, either formatted as ISO 8601
* yyyy-MM-dd'T'HH:mm:ssXXX
or as an int representing seconds since the epoch (like
* SOURCE_DATE_EPOCH).
*/
@Parameter(defaultValue = "${project.build.outputTimestamp}")
private String outputTimestamp;
/**
* Indicate if project workspace files with the same name should be used instead of the ones from the bundle.
*
* @since 3.3.0
*/
@Parameter(defaultValue = "false")
private boolean useProjectFiles;
/**
* Map of artifacts to supplemental project object models.
*/
private Map supplementModels;
/**
* Merges supplemental data model with artifact metadata. Useful when processing artifacts with
* incomplete POM metadata.
*/
private final ModelInheritanceAssembler inheritanceAssembler = new ModelInheritanceAssembler();
private VelocityEngine velocity;
protected final RepositorySystem repoSystem;
/**
* Filtering support, for local resources that override those in the remote bundle.
*/
private final MavenFileFilter fileFilter;
private final ResourceManager locator;
private final ProjectBuilder projectBuilder;
private final ArtifactHandlerManager artifactHandlerManager;
protected AbstractProcessRemoteResourcesMojo(
RepositorySystem repoSystem,
MavenFileFilter fileFilter,
ResourceManager locator,
ProjectBuilder projectBuilder,
ArtifactHandlerManager artifactHandlerManager) {
this.repoSystem = repoSystem;
this.fileFilter = fileFilter;
this.locator = locator;
this.projectBuilder = projectBuilder;
this.artifactHandlerManager = artifactHandlerManager;
}
@Override
public void execute() throws MojoExecutionException {
if (skip) {
getLog().info("Skipping remote resources execution.");
return;
}
if (encoding == null || encoding.isEmpty()) {
getLog().warn("File encoding has not been set, using platform encoding " + Charset.defaultCharset()
+ ", i.e. build is platform dependent!");
encoding = Charset.defaultCharset().name();
}
if (resolveScopes == null) {
resolveScopes = new String[] {
(this.includeScope == null || this.includeScope.isEmpty()) ? JavaScopes.TEST : this.includeScope
};
}
if (supplementalModels == null) {
File sups = new File(appendedResourcesDirectory, "supplemental-models.xml");
if (sups.exists()) {
try {
supplementalModels = new String[] {sups.toURI().toURL().toString()};
} catch (MalformedURLException e) {
// ignore
getLog().debug("URL issue with supplemental-models.xml: " + e);
}
}
}
configureLocator();
ClassLoader origLoader = Thread.currentThread().getContextClassLoader();
try {
validate();
List resourceBundleArtifacts = downloadBundles(resourceBundles);
supplementModels = loadSupplements(supplementalModels);
ClassLoader classLoader = initalizeClassloader(resourceBundleArtifacts);
Thread.currentThread().setContextClassLoader(classLoader);
velocity = new VelocityEngine();
velocity.setProperty("resource.loaders", "classpath");
velocity.setProperty("resource.loader.classpath.class", ClasspathResourceLoader.class.getName());
velocity.init();
VelocityContext context = buildVelocityContext();
processResourceBundles(classLoader, context);
if (outputDirectory.exists()) {
// ----------------------------------------------------------------------------
// Push our newly generated resources directory into the MavenProject so that
// these resources can be picked up by the process-resources phase.
// ----------------------------------------------------------------------------
Resource resource = new Resource();
resource.setDirectory(outputDirectory.getAbsolutePath());
// MRRESOURCES-61 handle main and test resources separately
if (attachToMain) {
project.getResources().add(resource);
}
if (attachToTest) {
project.getTestResources().add(resource);
}
// ----------------------------------------------------------------------------
// Write out archiver dot file
// ----------------------------------------------------------------------------
try {
File dotFile = new File(project.getBuild().getDirectory(), ".plxarc");
FileUtils.mkdir(dotFile.getParentFile().getAbsolutePath());
FileUtils.fileWrite(dotFile.getAbsolutePath(), outputDirectory.getName());
} catch (IOException e) {
throw new MojoExecutionException("Error creating dot file for archiving instructions.", e);
}
}
} finally {
Thread.currentThread().setContextClassLoader(origLoader);
}
}
private void configureLocator() throws MojoExecutionException {
if (supplementalModelArtifacts != null && !supplementalModelArtifacts.isEmpty()) {
List artifacts = downloadBundles(supplementalModelArtifacts);
for (File artifact : artifacts) {
if (artifact.isDirectory()) {
locator.addSearchPath(FileResourceLoader.ID, artifact.getAbsolutePath());
} else {
try {
locator.addSearchPath(
"jar", "jar:" + artifact.toURI().toURL().toExternalForm());
} catch (MalformedURLException e) {
throw new MojoExecutionException("Could not use jar " + artifact.getAbsolutePath(), e);
}
}
}
}
locator.addSearchPath(
FileResourceLoader.ID, project.getFile().getParentFile().getAbsolutePath());
if (appendedResourcesDirectory != null) {
locator.addSearchPath(FileResourceLoader.ID, appendedResourcesDirectory.getAbsolutePath());
}
locator.addSearchPath("url", "");
locator.setOutputDirectory(new File(project.getBuild().getDirectory()));
}
protected List getProjects() {
List projects = new ArrayList<>();
// add filters in well known order, least specific to most specific
FilterArtifacts filter = new FilterArtifacts();
Set artifacts = new LinkedHashSet<>(getAllDependencies());
if (this.excludeTransitive) {
filter.addFilter(new ProjectTransitivityFilter(getDirectDependencies(), true));
}
filter.addFilter(new ScopeFilter(this.includeScope, this.excludeScope));
filter.addFilter(new GroupIdFilter(this.includeGroupIds, this.excludeGroupIds));
filter.addFilter(new ArtifactIdFilter(this.includeArtifactIds, this.excludeArtifactIds));
// perform filtering
try {
artifacts = filter.filter(artifacts);
} catch (ArtifactFilterException e) {
throw new IllegalStateException(e.getMessage(), e);
}
getLog().debug("PROJECTS: " + artifacts);
for (Artifact artifact : artifacts) {
if (artifact.isSnapshot()) {
artifact.setVersion(artifact.getBaseVersion());
}
getLog().debug("Building project for " + artifact);
MavenProject p;
try {
ProjectBuildingRequest req = new DefaultProjectBuildingRequest()
.setValidationLevel(ModelBuildingRequest.VALIDATION_LEVEL_MINIMAL)
.setProcessPlugins(false)
.setRepositorySession(mavenSession.getRepositorySession())
.setSystemProperties(mavenSession.getSystemProperties())
.setUserProperties(mavenSession.getUserProperties())
.setLocalRepository(mavenSession.getLocalRepository())
.setRemoteRepositories(project.getRemoteArtifactRepositories());
ProjectBuildingResult res = projectBuilder.build(artifact, req);
p = res.getProject();
} catch (ProjectBuildingException e) {
getLog().warn("Invalid project model for artifact [" + artifact.getGroupId() + ":"
+ artifact.getArtifactId() + ":" + artifact.getVersion() + "]. "
+ "It will be ignored by the remote resources Mojo.");
continue;
}
String supplementKey = generateSupplementMapKey(
p.getModel().getGroupId(), p.getModel().getArtifactId());
if (supplementModels.containsKey(supplementKey)) {
Model mergedModel = mergeModels(p.getModel(), supplementModels.get(supplementKey));
MavenProject mergedProject = new MavenProject(mergedModel);
projects.add(mergedProject);
mergedProject.setArtifact(artifact);
mergedProject.setVersion(artifact.getVersion());
getLog().debug("Adding project with groupId [" + mergedProject.getGroupId() + "] (supplemented)");
} else {
projects.add(p);
getLog().debug("Adding project with groupId [" + p.getGroupId() + "]");
}
}
projects.sort(new ProjectComparator());
return projects;
}
/**
* Returns all the transitive hull of all the involved maven projects.
*/
protected abstract Set getAllDependencies();
/**
* Returns all the direct dependencies of all the involved maven projects.
*/
protected abstract Set getDirectDependencies();
protected Map> getProjectsSortedByOrganization(List projects) {
Map> organizations = new TreeMap<>(new OrganizationComparator());
List unknownOrganization = new ArrayList<>();
for (MavenProject p : projects) {
if (p.getOrganization() != null
&& StringUtils.isNotEmpty(p.getOrganization().getName())) {
List sortedProjects = organizations.get(p.getOrganization());
if (sortedProjects == null) {
sortedProjects = new ArrayList<>();
}
sortedProjects.add(p);
organizations.put(p.getOrganization(), sortedProjects);
} else {
unknownOrganization.add(p);
}
}
if (!unknownOrganization.isEmpty()) {
Organization unknownOrg = new Organization();
unknownOrg.setName("an unknown organization");
organizations.put(unknownOrg, unknownOrganization);
}
return organizations;
}
protected boolean copyResourceIfExists(
File outputFile, String bundleResourceName, VelocityContext context, String encoding)
throws IOException, MojoExecutionException {
for (Resource resource : project.getResources()) {
File resourceDirectory = new File(resource.getDirectory());
if (!resourceDirectory.exists()) {
continue;
}
// TODO - really should use the resource includes/excludes and name mapping
File source = new File(resourceDirectory, bundleResourceName);
File templateSource = new File(resourceDirectory, bundleResourceName + TEMPLATE_SUFFIX);
if (!source.exists() && templateSource.exists()) {
source = templateSource;
}
if (source.exists() && !source.equals(outputFile)) {
if (source == templateSource) {
getLog().debug("Use project resource '" + source + "' as resource with Velocity");
try (CachingOutputStream os = new CachingOutputStream(outputFile);
Writer writer = getWriter(encoding, os);
Reader reader = getReader(encoding, source)) {
velocity.evaluate(context, writer, "", reader);
} catch (ParseErrorException | MethodInvocationException | ResourceNotFoundException e) {
throw new MojoExecutionException("Error rendering velocity resource: " + source, e);
}
} else if (resource.isFiltering()) {
getLog().debug("Use project resource '" + source + "' as resource with filtering");
MavenFileFilterRequest req = setupRequest(resource, source, outputFile);
try {
fileFilter.copyFile(req);
} catch (MavenFilteringException e) {
throw new MojoExecutionException("Error filtering resource: " + source, e);
}
} else {
getLog().debug("Use project resource '" + source + "' as resource");
FilteringUtils.copyFile(source, outputFile, null, null);
}
// exclude the original (so eclipse doesn't complain about duplicate resources)
resource.addExclude(bundleResourceName);
return true;
}
}
return false;
}
private boolean copyProjectRootIfExists(File outputFile, String bundleResourceName) throws IOException {
if (!useProjectFiles) {
return false;
}
File source = new File(project.getBasedir(), bundleResourceName);
if (source.exists()) {
getLog().debug("Use project file '" + source + "' as resource");
FilteringUtils.copyFile(source, outputFile, null, null);
return true;
}
return false;
}
private Reader getReader(String readerEncoding, File file) throws IOException {
return Files.newBufferedReader(
file.toPath(), Charset.forName(readerEncoding != null ? readerEncoding : encoding));
}
private Writer getWriter(String writerEncoding, OutputStream outputStream) throws IOException {
return new OutputStreamWriter(outputStream, writerEncoding != null ? writerEncoding : encoding);
}
private MavenFileFilterRequest setupRequest(Resource resource, File source, File file) {
MavenFileFilterRequest req = new MavenFileFilterRequest();
req.setFrom(source);
req.setTo(file);
req.setFiltering(resource.isFiltering());
req.setMavenProject(project);
req.setMavenSession(mavenSession);
req.setInjectProjectBuildFilters(true);
req.setEncoding(encoding);
if (filterDelimiters != null && !filterDelimiters.isEmpty()) {
LinkedHashSet delims = new LinkedHashSet<>();
if (useDefaultFilterDelimiters) {
delims.addAll(req.getDelimiters());
}
for (String delim : filterDelimiters) {
if (delim == null) {
delims.add("${*}");
} else {
delims.add(delim);
}
}
req.setDelimiters(delims);
}
return req;
}
protected void validate() throws MojoExecutionException {
int bundleCount = 1;
for (String artifactDescriptor : resourceBundles) {
// groupId:artifactId:version, groupId:artifactId:version:type
// or groupId:artifactId:version:type:classifier
String[] s = StringUtils.split(artifactDescriptor, ":");
if (s.length < 3 || s.length > 5) {
String position;
if (bundleCount == 1) {
position = "1st";
} else if (bundleCount == 2) {
position = "2nd";
} else if (bundleCount == 3) {
position = "3rd";
} else {
position = bundleCount + "th";
}
throw new MojoExecutionException("The " + position
+ " resource bundle configured must specify a groupId, artifactId, "
+ " version and, optionally, type and classifier for a remote resource bundle. "
+ "Must be of the form groupId:artifactId:version , "
+ "groupId:artifactId:version:type or "
+ "groupId:artifactId:version:type:classifier ");
}
bundleCount++;
}
}
private static final String KEY_PROJECTS = "projects";
private static final String KEY_PROJECTS_ORGS = "projectsSortedByOrganization";
protected VelocityContext buildVelocityContext() {
Map contextProperties = new HashMap<>(properties);
if (includeProjectProperties) {
final Properties projectProperties = project.getProperties();
for (String key : projectProperties.stringPropertyNames()) {
contextProperties.put(key, projectProperties.getProperty(key));
}
}
// the following properties are expensive to calculate, so we provide them lazily
VelocityContext context = new VelocityContext(contextProperties) {
@Override
public Object internalGet(String key) {
Object result = super.internalGet(key);
if (result == null && key != null && key.startsWith(KEY_PROJECTS) && containsKey(key)) {
// calculate and put projects* properties
List projects = getProjects();
put(KEY_PROJECTS, projects);
put(KEY_PROJECTS_ORGS, getProjectsSortedByOrganization(projects));
return super.internalGet(key);
}
return result;
}
};
// to have a consistent getKeys()/containsKey() behaviour, keys must be present from the start
context.put(KEY_PROJECTS, null);
context.put(KEY_PROJECTS_ORGS, null);
// the following properties are cheap to calculate, so we provide them eagerly
String inceptionYear = project.getInceptionYear();
// Reproducible Builds: try to use reproducible output timestamp
String year = MavenArchiver.parseBuildOutputTimestamp(outputTimestamp)
.orElseGet(Instant::now)
.atZone(ZoneId.of("UTC+10"))
.format(DateTimeFormatter.ofPattern("yyyy"));
if (inceptionYear == null || inceptionYear.isEmpty()) {
if (getLog().isDebugEnabled()) {
getLog().debug("inceptionYear not specified, defaulting to " + year);
}
inceptionYear = year;
}
context.put("project", project);
context.put("presentYear", year);
context.put("locator", locator);
if (inceptionYear.equals(year)) {
context.put("projectTimespan", year);
} else {
context.put("projectTimespan", inceptionYear + "-" + year);
}
return context;
}
private List downloadBundles(List bundles) throws MojoExecutionException {
List bundleArtifacts = new ArrayList<>();
for (String artifactDescriptor : bundles) {
getLog().info("Preparing remote bundle " + artifactDescriptor);
// groupId:artifactId:version[:type[:classifier]]
String[] s = artifactDescriptor.split(":");
File artifactFile = null;
// check if the artifact is part of the reactor
if (mavenSession != null) {
List list = mavenSession.getProjects();
for (MavenProject p : list) {
if (s[0].equals(p.getGroupId()) && s[1].equals(p.getArtifactId()) && s[2].equals(p.getVersion())) {
if (s.length >= 4 && "test-jar".equals(s[3])) {
artifactFile = new File(p.getBuild().getTestOutputDirectory());
} else {
artifactFile = new File(p.getBuild().getOutputDirectory());
}
}
}
}
if (artifactFile == null || !artifactFile.exists()) {
String g = s[0];
String a = s[1];
String v = s[2];
String type = s.length >= 4 ? s[3] : "jar";
ArtifactType artifactType =
RepositoryUtils.newArtifactType(type, artifactHandlerManager.getArtifactHandler(type));
String classifier = s.length == 5 ? s[4] : artifactType.getClassifier();
DefaultArtifact artifact =
new DefaultArtifact(g, a, classifier, artifactType.getExtension(), v, artifactType);
try {
ArtifactRequest request =
new ArtifactRequest(artifact, project.getRemoteProjectRepositories(), "remote-resources");
ArtifactResult result = repoSystem.resolveArtifact(mavenSession.getRepositorySession(), request);
artifactFile = result.getArtifact().getFile();
} catch (ArtifactResolutionException e) {
throw new MojoExecutionException("Error processing remote resources", e);
}
}
bundleArtifacts.add(artifactFile);
}
return bundleArtifacts;
}
private ClassLoader initalizeClassloader(List artifacts) throws MojoExecutionException {
RemoteResourcesClassLoader cl = new RemoteResourcesClassLoader(null);
try {
for (File artifact : artifacts) {
cl.addURL(artifact.toURI().toURL());
}
return cl;
} catch (MalformedURLException e) {
throw new MojoExecutionException("Unable to configure resources classloader: " + e.getMessage(), e);
}
}
protected void processResourceBundles(ClassLoader classLoader, VelocityContext context)
throws MojoExecutionException {
List> remoteResources = new ArrayList<>();
int bundleCount = 0;
int resourceCount = 0;
// list remote resources form bundles
try {
RemoteResourcesBundleXpp3Reader bundleReader = new RemoteResourcesBundleXpp3Reader();
for (Enumeration e = classLoader.getResources(BundleRemoteResourcesMojo.RESOURCES_MANIFEST);
e.hasMoreElements(); ) {
URL url = e.nextElement();
bundleCount++;
getLog().debug("processResourceBundle on bundle#" + bundleCount + " " + url);
RemoteResourcesBundle bundle;
try (InputStream in = url.openStream()) {
bundle = bundleReader.read(in);
}
verifyRequiredProperties(bundle, url);
int n = 0;
for (String bundleResource : bundle.getRemoteResources()) {
n++;
resourceCount++;
getLog().debug("bundle#" + bundleCount + " resource#" + n + " " + bundleResource);
remoteResources.add(new AbstractMap.SimpleEntry<>(bundleResource, bundle));
}
}
} catch (IOException ioe) {
throw new MojoExecutionException("Error finding remote resources manifests", ioe);
} catch (XmlPullParserException xppe) {
throw new MojoExecutionException("Error parsing remote resource bundle descriptor.", xppe);
}
getLog().info("Copying " + resourceCount + " resource" + ((resourceCount > 1) ? "s" : "") + " from "
+ bundleCount + " bundle" + ((bundleCount > 1) ? "s" : "") + ".");
String velocityResource = null;
try {
for (Map.Entry entry : remoteResources) {
String bundleResource = entry.getKey();
RemoteResourcesBundle bundle = entry.getValue();
String projectResource = bundleResource;
boolean doVelocity = false;
if (projectResource.endsWith(TEMPLATE_SUFFIX)) {
projectResource = projectResource.substring(0, projectResource.length() - 3);
velocityResource = bundleResource;
doVelocity = true;
}
// Don't overwrite resource that are already being provided.
File outputFile = new File(outputDirectory, projectResource);
FileUtils.mkdir(outputFile.getParentFile().getAbsolutePath());
// resource exists in project resources
if (copyResourceIfExists(outputFile, projectResource, context, bundle.getSourceEncoding())) {
continue;
}
if (copyProjectRootIfExists(outputFile, projectResource)) {
continue;
}
if (doVelocity) {
String bundleEncoding = bundle.getSourceEncoding();
if (bundleEncoding == null) {
bundleEncoding = encoding;
}
try (CachingOutputStream os = new CachingOutputStream(outputFile);
Writer writer = getWriter(bundleEncoding, os)) {
velocity.mergeTemplate(bundleResource, bundleEncoding, context, writer);
}
} else {
URL bundleResourceUrl = classLoader.getResource(bundleResource);
if (bundleResourceUrl != null) {
FileUtils.copyURLToFile(bundleResourceUrl, outputFile);
}
}
File appendedResourceFile = new File(appendedResourcesDirectory, projectResource);
File appendedVmResourceFile = new File(appendedResourcesDirectory, projectResource + ".vm");
if (appendedResourceFile.exists()) {
getLog().info("Copying appended resource: " + projectResource);
try (InputStream in = Files.newInputStream(appendedResourceFile.toPath());
OutputStream out = new FileOutputStream(outputFile, true)) {
IOUtil.copy(in, out);
}
} else if (appendedVmResourceFile.exists()) {
getLog().info("Filtering appended resource: " + projectResource + ".vm");
try (CachingOutputStream os = new CachingOutputStream(outputFile);
Reader reader = getReader(bundle.getSourceEncoding(), appendedVmResourceFile);
Writer writer = getWriter(bundle.getSourceEncoding(), os)) {
Velocity.init();
Velocity.evaluate(context, writer, "remote-resources", reader);
}
}
}
} catch (IOException ioe) {
throw new MojoExecutionException("Error reading remote resource", ioe);
} catch (VelocityException e) {
throw new MojoExecutionException("Error rendering Velocity resource '" + velocityResource + "'", e);
}
}
private void verifyRequiredProperties(RemoteResourcesBundle bundle, URL url) throws MojoExecutionException {
if (bundle.getRequiredProjectProperties() == null
|| bundle.getRequiredProjectProperties().isEmpty()) {
return;
}
for (String requiredProperty : bundle.getRequiredProjectProperties()) {
if (!project.getProperties().containsKey(requiredProperty)) {
throw new MojoExecutionException(
"Required project property: '" + requiredProperty + "' is not present for bundle: " + url);
}
}
}
protected Model getSupplement(Xpp3Dom supplementModelXml) throws MojoExecutionException {
MavenXpp3Reader modelReader = new MavenXpp3Reader();
Model model = null;
try {
model = modelReader.read(new StringReader(supplementModelXml.toString()));
String groupId = model.getGroupId();
String artifactId = model.getArtifactId();
if (groupId == null || groupId.trim().isEmpty()) {
throw new MojoExecutionException(
"Supplemental project XML " + "requires that a element be present.");
}
if (artifactId == null || artifactId.trim().isEmpty()) {
throw new MojoExecutionException(
"Supplemental project XML " + "requires that a element be present.");
}
} catch (IOException e) {
getLog().warn("Unable to read supplemental XML: " + e.getMessage(), e);
} catch (XmlPullParserException e) {
getLog().warn("Unable to parse supplemental XML: " + e.getMessage(), e);
}
return model;
}
protected Model mergeModels(Model parent, Model child) {
inheritanceAssembler.assembleModelInheritance(child, parent);
return child;
}
private static String generateSupplementMapKey(String groupId, String artifactId) {
return groupId.trim() + ":" + artifactId.trim();
}
private Map loadSupplements(String[] models) throws MojoExecutionException {
if (models == null) {
getLog().debug("Supplemental data models won't be loaded. No models specified.");
return Collections.emptyMap();
}
List supplements = new ArrayList<>();
for (String set : models) {
getLog().debug("Preparing ruleset: " + set);
try {
File f = locator.getResourceAsFile(set, getLocationTemp(set));
if (null == f || !f.exists()) {
throw new MojoExecutionException("Cold not resolve " + set);
}
if (!f.canRead()) {
throw new MojoExecutionException("Supplemental data models won't be loaded. " + "File "
+ f.getAbsolutePath() + " cannot be read, check permissions on the file.");
}
getLog().debug("Loading supplemental models from " + f.getAbsolutePath());
SupplementalDataModelXpp3Reader reader = new SupplementalDataModelXpp3Reader();
SupplementalDataModel supplementalModel = reader.read(new FileReader(f));
supplements.addAll(supplementalModel.getSupplement());
} catch (Exception e) {
String msg = "Error loading supplemental data models: " + e.getMessage();
getLog().error(msg, e);
throw new MojoExecutionException(msg, e);
}
}
getLog().debug("Loading supplements complete.");
Map supplementMap = new HashMap<>();
for (Supplement sd : supplements) {
Xpp3Dom dom = (Xpp3Dom) sd.getProject();
Model m = getSupplement(dom);
supplementMap.put(generateSupplementMapKey(m.getGroupId(), m.getArtifactId()), m);
}
return supplementMap;
}
/**
* Convenience method to get the location of the specified file name.
*
* @param name the name of the file whose location is to be resolved
* @return a String that contains the absolute file name of the file
*/
private String getLocationTemp(String name) {
String loc = name;
if (loc.indexOf('/') != -1) {
loc = loc.substring(loc.lastIndexOf('/') + 1);
}
if (loc.indexOf('\\') != -1) {
loc = loc.substring(loc.lastIndexOf('\\') + 1);
}
getLog().debug("Before: " + name + " After: " + loc);
return loc;
}
static class OrganizationComparator implements Comparator {
@Override
public int compare(Organization org1, Organization org2) {
int i = compareStrings(org1.getName(), org2.getName());
if (i == 0) {
i = compareStrings(org1.getUrl(), org2.getUrl());
}
return i;
}
private int compareStrings(String s1, String s2) {
if (s1 == null && s2 == null) {
return 0;
} else if (s1 == null) {
return 1;
} else if (s2 == null) {
return -1;
}
return s1.compareToIgnoreCase(s2);
}
}
static class ProjectComparator implements Comparator {
@Override
public int compare(MavenProject p1, MavenProject p2) {
return p1.getArtifact().compareTo(p2.getArtifact());
}
}
}