blob: aac66e674e22c4343c3560c12f766521bba5d1f0 [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.maven.project;
import javax.inject.Inject;
import javax.inject.Named;
import javax.inject.Singleton;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.HashMap;
import java.util.HashSet;
import java.util.LinkedHashSet;
import java.util.List;
import java.util.Map;
import java.util.Set;
import org.apache.maven.RepositoryUtils;
import org.apache.maven.api.xml.XmlNode;
import org.apache.maven.artifact.Artifact;
import org.apache.maven.artifact.InvalidRepositoryException;
import org.apache.maven.artifact.repository.ArtifactRepository;
import org.apache.maven.bridge.MavenRepositorySystem;
import org.apache.maven.classrealm.ClassRealmManager;
import org.apache.maven.model.Build;
import org.apache.maven.model.Extension;
import org.apache.maven.model.Model;
import org.apache.maven.model.Plugin;
import org.apache.maven.model.Repository;
import org.apache.maven.plugin.ExtensionRealmCache;
import org.apache.maven.plugin.MavenPluginManager;
import org.apache.maven.plugin.PluginManagerException;
import org.apache.maven.plugin.PluginResolutionException;
import org.apache.maven.plugin.version.PluginVersionResolutionException;
import org.codehaus.plexus.PlexusContainer;
import org.codehaus.plexus.classworlds.realm.ClassRealm;
import org.codehaus.plexus.util.xml.Xpp3Dom;
import org.eclipse.aether.graph.DependencyFilter;
import org.eclipse.aether.util.filter.ExclusionsDependencyFilter;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
/**
* Assists the project builder. <strong>Warning:</strong> This is an internal utility class that is only public for
* technical reasons, it is not part of the public API. In particular, this class can be changed or deleted without
* prior notice.
*
*/
@Named
@Singleton
public class DefaultProjectBuildingHelper implements ProjectBuildingHelper {
private final Logger logger = LoggerFactory.getLogger(getClass());
private final PlexusContainer container; // TODO not used? Then remove
private final ClassRealmManager classRealmManager;
private final ProjectRealmCache projectRealmCache;
private final MavenRepositorySystem repositorySystem;
private final MavenPluginManager pluginManager;
@Inject
public DefaultProjectBuildingHelper(
PlexusContainer container,
ClassRealmManager classRealmManager,
ProjectRealmCache projectRealmCache,
MavenRepositorySystem repositorySystem,
MavenPluginManager pluginManager) {
this.container = container;
this.classRealmManager = classRealmManager;
this.projectRealmCache = projectRealmCache;
this.repositorySystem = repositorySystem;
this.pluginManager = pluginManager;
}
public List<ArtifactRepository> createArtifactRepositories(
List<Repository> pomRepositories,
List<ArtifactRepository> externalRepositories,
ProjectBuildingRequest request)
throws InvalidRepositoryException {
List<ArtifactRepository> internalRepositories = new ArrayList<>();
for (Repository repository : pomRepositories) {
internalRepositories.add(MavenRepositorySystem.buildArtifactRepository(repository));
}
repositorySystem.injectMirror(request.getRepositorySession(), internalRepositories);
repositorySystem.injectProxy(request.getRepositorySession(), internalRepositories);
repositorySystem.injectAuthentication(request.getRepositorySession(), internalRepositories);
List<ArtifactRepository> dominantRepositories;
List<ArtifactRepository> recessiveRepositories;
if (ProjectBuildingRequest.RepositoryMerging.REQUEST_DOMINANT.equals(request.getRepositoryMerging())) {
dominantRepositories = externalRepositories;
recessiveRepositories = internalRepositories;
} else {
dominantRepositories = internalRepositories;
recessiveRepositories = externalRepositories;
}
List<ArtifactRepository> artifactRepositories = new ArrayList<>();
Collection<String> repoIds = new HashSet<>();
if (dominantRepositories != null) {
for (ArtifactRepository repository : dominantRepositories) {
repoIds.add(repository.getId());
artifactRepositories.add(repository);
}
}
if (recessiveRepositories != null) {
for (ArtifactRepository repository : recessiveRepositories) {
if (repoIds.add(repository.getId())) {
artifactRepositories.add(repository);
}
}
}
artifactRepositories = repositorySystem.getEffectiveRepositories(artifactRepositories);
return artifactRepositories;
}
public synchronized ProjectRealmCache.CacheRecord createProjectRealm(
MavenProject project, Model model, ProjectBuildingRequest request)
throws PluginResolutionException, PluginVersionResolutionException, PluginManagerException {
ClassRealm projectRealm;
List<Plugin> extensionPlugins = new ArrayList<>();
Build build = model.getBuild();
if (build != null) {
for (Extension extension : build.getExtensions()) {
Plugin plugin = new Plugin();
plugin.setGroupId(extension.getGroupId());
plugin.setArtifactId(extension.getArtifactId());
plugin.setVersion(extension.getVersion());
XmlNode configuration = extension.getDelegate().getConfiguration();
if (configuration != null) {
plugin.setConfiguration(new Xpp3Dom(configuration));
}
extensionPlugins.add(plugin);
}
for (Plugin plugin : build.getPlugins()) {
if (plugin.isExtensions()) {
extensionPlugins.add(plugin);
}
}
}
if (extensionPlugins.isEmpty()) {
if (logger.isDebugEnabled()) {
logger.debug("Extension realms for project " + model.getId() + ": (none)");
}
return new ProjectRealmCache.CacheRecord(null, null);
}
List<ClassRealm> extensionRealms = new ArrayList<>();
Map<ClassRealm, List<String>> exportedPackages = new HashMap<>();
Map<ClassRealm, List<String>> exportedArtifacts = new HashMap<>();
List<Artifact> publicArtifacts = new ArrayList<>();
for (Plugin plugin : extensionPlugins) {
ExtensionRealmCache.CacheRecord recordRealm =
pluginManager.setupExtensionsRealm(project, plugin, request.getRepositorySession());
final ClassRealm extensionRealm = recordRealm.getRealm();
final ExtensionDescriptor extensionDescriptor = recordRealm.getDescriptor();
final List<Artifact> artifacts = recordRealm.getArtifacts();
extensionRealms.add(extensionRealm);
if (extensionDescriptor != null) {
exportedPackages.put(extensionRealm, extensionDescriptor.getExportedPackages());
exportedArtifacts.put(extensionRealm, extensionDescriptor.getExportedArtifacts());
}
if (!plugin.isExtensions()
&& artifacts.size() == 1
&& artifacts.get(0).getFile() != null) {
/*
* This is purely for backward-compat with 2.x where <extensions> consisting of a single artifact where
* loaded into the core and hence available to plugins, in contrast to bigger extensions that were
* loaded into a dedicated realm which is invisible to plugins (MNG-2749).
*/
publicArtifacts.addAll(artifacts);
}
}
if (logger.isDebugEnabled()) {
logger.debug("Extension realms for project " + model.getId() + ": " + extensionRealms);
}
ProjectRealmCache.Key projectRealmKey = projectRealmCache.createKey(extensionRealms);
ProjectRealmCache.CacheRecord record = projectRealmCache.get(projectRealmKey);
if (record == null) {
projectRealm = classRealmManager.createProjectRealm(model, toAetherArtifacts(publicArtifacts));
Set<String> exclusions = new LinkedHashSet<>();
for (ClassRealm extensionRealm : extensionRealms) {
List<String> excludes = exportedArtifacts.get(extensionRealm);
if (excludes != null) {
exclusions.addAll(excludes);
}
List<String> exports = exportedPackages.get(extensionRealm);
if (exports == null || exports.isEmpty()) {
/*
* Most existing extensions don't define exported packages, i.e. no classes are to be exposed to
* plugins, yet the components provided by the extension (e.g. artifact handlers) must be
* accessible, i.e. we still must import the extension realm into the project realm.
*/
exports = Arrays.asList(extensionRealm.getId());
}
for (String export : exports) {
projectRealm.importFrom(extensionRealm, export);
}
}
DependencyFilter extensionArtifactFilter = null;
if (!exclusions.isEmpty()) {
extensionArtifactFilter = new ExclusionsDependencyFilter(exclusions);
}
record = projectRealmCache.put(projectRealmKey, projectRealm, extensionArtifactFilter);
}
projectRealmCache.register(project, projectRealmKey, record);
return record;
}
public void selectProjectRealm(MavenProject project) {
ClassLoader projectRealm = project.getClassRealm();
if (projectRealm == null) {
projectRealm = classRealmManager.getCoreRealm();
}
Thread.currentThread().setContextClassLoader(projectRealm);
}
private List<org.eclipse.aether.artifact.Artifact> toAetherArtifacts(final List<Artifact> pluginArtifacts) {
return new ArrayList<>(RepositoryUtils.toArtifacts(pluginArtifacts));
}
}