org.apache.maven.DefaultMaven Maven / Gradle / Ivy
/*
* 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;
import java.io.File;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.Collections;
import java.util.Date;
import java.util.HashSet;
import java.util.LinkedHashMap;
import java.util.LinkedHashSet;
import java.util.List;
import java.util.Map;
import java.util.Set;
import org.apache.maven.artifact.ArtifactUtils;
import org.apache.maven.execution.DefaultMavenExecutionResult;
import org.apache.maven.execution.ExecutionEvent;
import org.apache.maven.execution.MavenExecutionRequest;
import org.apache.maven.execution.MavenExecutionResult;
import org.apache.maven.execution.MavenSession;
import org.apache.maven.execution.ProjectDependencyGraph;
import org.apache.maven.graph.GraphBuilder;
import org.apache.maven.internal.aether.DefaultRepositorySystemSessionFactory;
import org.apache.maven.internal.aether.MavenChainedWorkspaceReader;
import org.apache.maven.lifecycle.internal.ExecutionEventCatapult;
import org.apache.maven.lifecycle.internal.LifecycleStarter;
import org.apache.maven.model.Prerequisites;
import org.apache.maven.model.building.ModelProblem;
import org.apache.maven.model.building.Result;
import org.apache.maven.plugin.LegacySupport;
import org.apache.maven.project.MavenProject;
import org.apache.maven.project.ProjectBuilder;
import org.apache.maven.repository.LocalRepositoryNotAccessibleException;
import org.apache.maven.session.scope.internal.SessionScope;
import org.codehaus.plexus.PlexusContainer;
import org.codehaus.plexus.component.annotations.Component;
import org.codehaus.plexus.component.annotations.Requirement;
import org.codehaus.plexus.component.repository.exception.ComponentLookupException;
import org.codehaus.plexus.logging.Logger;
import org.eclipse.aether.DefaultRepositorySystemSession;
import org.eclipse.aether.RepositorySystemSession;
import org.eclipse.aether.repository.WorkspaceReader;
/**
* @author Jason van Zyl
*/
@Component(role = Maven.class)
public class DefaultMaven implements Maven {
@Requirement
private Logger logger;
@Requirement
protected ProjectBuilder projectBuilder;
@Requirement
private LifecycleStarter lifecycleStarter;
@Requirement
protected PlexusContainer container;
@Requirement
private ExecutionEventCatapult eventCatapult;
@Requirement
private LegacySupport legacySupport;
@Requirement
private SessionScope sessionScope;
@Requirement
private DefaultRepositorySystemSessionFactory repositorySessionFactory;
@Requirement(hint = GraphBuilder.HINT)
private GraphBuilder graphBuilder;
@Override
public MavenExecutionResult execute(MavenExecutionRequest request) {
MavenExecutionResult result;
try {
result = doExecute(request);
} catch (OutOfMemoryError e) {
result = addExceptionToResult(new DefaultMavenExecutionResult(), e);
} catch (RuntimeException e) {
// TODO Hack to make the cycle detection the same for the new graph builder
if (e.getCause() instanceof ProjectCycleException) {
result = addExceptionToResult(new DefaultMavenExecutionResult(), e.getCause());
} else {
result = addExceptionToResult(
new DefaultMavenExecutionResult(), new InternalErrorException("Internal error: " + e, e));
}
} finally {
legacySupport.setSession(null);
}
return result;
}
//
// 1) Setup initial properties.
//
// 2) Validate local repository directory is accessible.
//
// 3) Create RepositorySystemSession.
//
// 4) Create MavenSession.
//
// 5) Execute AbstractLifecycleParticipant.afterSessionStart(session)
//
// 6) Get reactor projects looking for general POM errors
//
// 7) Create ProjectDependencyGraph using trimming which takes into account --projects and reactor mode.
// This ensures that the projects passed into the ReactorReader are only those specified.
//
// 8) Create ReactorReader with the getProjectMap( projects ). NOTE that getProjectMap(projects) is the code that
// checks for duplicate projects definitions in the build. Ideally this type of duplicate checking should be
// part of getting the reactor projects in 6). The duplicate checking is conflated with getProjectMap(projects).
//
// 9) Execute AbstractLifecycleParticipant.afterProjectsRead(session)
//
// 10) Create ProjectDependencyGraph without trimming (as trimming was done in 7). A new topological sort is
// required after the execution of 9) as the AbstractLifecycleParticipants are free to mutate the MavenProject
// instances, which may change dependencies which can, in turn, affect the build order.
//
// 11) Execute LifecycleStarter.start()
//
@SuppressWarnings("checkstyle:methodlength")
private MavenExecutionResult doExecute(MavenExecutionRequest request) {
request.setStartTime(new Date());
MavenExecutionResult result = new DefaultMavenExecutionResult();
try {
validateLocalRepository(request);
} catch (LocalRepositoryNotAccessibleException e) {
return addExceptionToResult(result, e);
}
//
// We enter the session scope right after the MavenSession creation and before any of the
// AbstractLifecycleParticipant lookups
// so that @SessionScoped components can be @Injected into AbstractLifecycleParticipants.
//
sessionScope.enter();
try {
DefaultRepositorySystemSession repoSession = (DefaultRepositorySystemSession) newRepositorySession(request);
MavenSession session = new MavenSession(container, repoSession, request, result);
sessionScope.seed(MavenSession.class, session);
legacySupport.setSession(session);
return doExecute(request, session, result, repoSession);
} finally {
sessionScope.exit();
}
}
private MavenExecutionResult doExecute(
MavenExecutionRequest request,
MavenSession session,
MavenExecutionResult result,
DefaultRepositorySystemSession repoSession) {
try {
// CHECKSTYLE_OFF: LineLength
for (AbstractMavenLifecycleParticipant listener :
getLifecycleParticipants(Collections.emptyList())) {
listener.afterSessionStart(session);
}
// CHECKSTYLE_ON: LineLength
} catch (MavenExecutionException e) {
return addExceptionToResult(result, e);
}
eventCatapult.fire(ExecutionEvent.Type.ProjectDiscoveryStarted, session, null);
Result extends ProjectDependencyGraph> graphResult = buildGraph(session, result);
if (graphResult.hasErrors()) {
return addExceptionToResult(
result, graphResult.getProblems().iterator().next().getException());
}
try {
session.setProjectMap(getProjectMap(session.getProjects()));
} catch (DuplicateProjectException e) {
return addExceptionToResult(result, e);
}
try {
setupWorkspaceReader(session, repoSession);
} catch (ComponentLookupException e) {
return addExceptionToResult(result, e);
}
repoSession.setReadOnly();
ClassLoader originalClassLoader = Thread.currentThread().getContextClassLoader();
try {
for (AbstractMavenLifecycleParticipant listener : getLifecycleParticipants(session.getProjects())) {
Thread.currentThread().setContextClassLoader(listener.getClass().getClassLoader());
listener.afterProjectsRead(session);
}
} catch (MavenExecutionException e) {
return addExceptionToResult(result, e);
} finally {
Thread.currentThread().setContextClassLoader(originalClassLoader);
}
//
// The projects need to be topologically after the participants have run their afterProjectsRead(session)
// because the participant is free to change the dependencies of a project which can potentially change the
// topological order of the projects, and therefore can potentially change the build order.
//
// Note that participants may affect the topological order of the projects but it is
// not expected that a participant will add or remove projects from the session.
//
graphResult = buildGraph(session, result);
if (graphResult.hasErrors()) {
return addExceptionToResult(
result, graphResult.getProblems().iterator().next().getException());
}
try {
if (result.hasExceptions()) {
return result;
}
result.setTopologicallySortedProjects(session.getProjects());
result.setProject(session.getTopLevelProject());
validatePrerequisitesForNonMavenPluginProjects(session.getProjects());
validateActivatedProfiles(
session.getProjects(), request.getActiveProfiles(), request.getInactiveProfiles());
lifecycleStarter.execute(session);
validateActivatedProfiles(
session.getProjects(), request.getActiveProfiles(), request.getInactiveProfiles());
if (session.getResult().hasExceptions()) {
return addExceptionToResult(
result, session.getResult().getExceptions().get(0));
}
} finally {
try {
afterSessionEnd(session.getProjects(), session);
} catch (MavenExecutionException e) {
return addExceptionToResult(result, e);
}
}
return result;
}
private void setupWorkspaceReader(MavenSession session, DefaultRepositorySystemSession repoSession)
throws ComponentLookupException {
// Desired order of precedence for workspace readers before querying the local artifact repositories
Set workspaceReaders = new LinkedHashSet<>();
// 1) Reactor workspace reader
workspaceReaders.add(container.lookup(WorkspaceReader.class, ReactorReader.HINT));
// 2) Repository system session-scoped workspace reader
WorkspaceReader repoWorkspaceReader = repoSession.getWorkspaceReader();
if (repoWorkspaceReader != null) {
workspaceReaders.add(repoWorkspaceReader);
}
// 3) .. n) Project-scoped workspace readers
workspaceReaders.addAll(getProjectScopedExtensionComponents(session.getProjects(), WorkspaceReader.class));
repoSession.setWorkspaceReader(MavenChainedWorkspaceReader.of(workspaceReaders));
}
private void afterSessionEnd(Collection projects, MavenSession session)
throws MavenExecutionException {
ClassLoader originalClassLoader = Thread.currentThread().getContextClassLoader();
try {
for (AbstractMavenLifecycleParticipant listener : getLifecycleParticipants(projects)) {
Thread.currentThread().setContextClassLoader(listener.getClass().getClassLoader());
listener.afterSessionEnd(session);
}
} finally {
Thread.currentThread().setContextClassLoader(originalClassLoader);
}
}
public RepositorySystemSession newRepositorySession(MavenExecutionRequest request) {
return repositorySessionFactory.newRepositorySession(request);
}
private void validateLocalRepository(MavenExecutionRequest request) throws LocalRepositoryNotAccessibleException {
File localRepoDir = request.getLocalRepositoryPath();
logger.debug("Using local repository at " + localRepoDir);
localRepoDir.mkdirs();
if (!localRepoDir.isDirectory()) {
throw new LocalRepositoryNotAccessibleException("Could not create local repository at " + localRepoDir);
}
}
private Collection getLifecycleParticipants(Collection projects) {
Collection lifecycleListeners = new LinkedHashSet<>();
try {
lifecycleListeners.addAll(container.lookupList(AbstractMavenLifecycleParticipant.class));
} catch (ComponentLookupException e) {
// this is just silly, lookupList should return an empty list!
logger.warn("Failed to lookup lifecycle participants: " + e.getMessage());
}
lifecycleListeners.addAll(
getProjectScopedExtensionComponents(projects, AbstractMavenLifecycleParticipant.class));
return lifecycleListeners;
}
protected Collection getProjectScopedExtensionComponents(Collection projects, Class role) {
Collection foundComponents = new LinkedHashSet<>();
Collection scannedRealms = new HashSet<>();
Thread currentThread = Thread.currentThread();
ClassLoader originalContextClassLoader = currentThread.getContextClassLoader();
try {
for (MavenProject project : projects) {
ClassLoader projectRealm = project.getClassRealm();
if (projectRealm != null && scannedRealms.add(projectRealm)) {
currentThread.setContextClassLoader(projectRealm);
try {
foundComponents.addAll(container.lookupList(role));
} catch (ComponentLookupException e) {
// this is just silly, lookupList should return an empty list!
logger.warn("Failed to lookup " + role + ": " + e.getMessage());
}
}
}
return foundComponents;
} finally {
currentThread.setContextClassLoader(originalContextClassLoader);
}
}
private MavenExecutionResult addExceptionToResult(MavenExecutionResult result, Throwable e) {
if (!result.getExceptions().contains(e)) {
result.addException(e);
}
return result;
}
private void validatePrerequisitesForNonMavenPluginProjects(List projects) {
for (MavenProject mavenProject : projects) {
if (!"maven-plugin".equals(mavenProject.getPackaging())) {
Prerequisites prerequisites = mavenProject.getPrerequisites();
if (prerequisites != null && prerequisites.getMaven() != null) {
logger.warn("The project " + mavenProject.getId() + " uses prerequisites"
+ " which is only intended for maven-plugin projects "
+ "but not for non maven-plugin projects. "
+ "For such purposes you should use the maven-enforcer-plugin. "
+ "See https://maven.apache.org/enforcer/enforcer-rules/requireMavenVersion.html");
}
}
}
}
private void validateActivatedProfiles(
List projects, List activeProfileIds, List inactiveProfileIds) {
Collection notActivatedProfileIds = new LinkedHashSet<>(activeProfileIds);
for (MavenProject project : projects) {
for (List profileIds : project.getInjectedProfileIds().values()) {
notActivatedProfileIds.removeAll(profileIds);
}
}
notActivatedProfileIds.removeAll(inactiveProfileIds);
for (String notActivatedProfileId : notActivatedProfileIds) {
logger.warn("The requested profile \"" + notActivatedProfileId
+ "\" could not be activated because it does not exist.");
}
}
private Map getProjectMap(Collection projects)
throws DuplicateProjectException {
Map index = new LinkedHashMap<>();
Map> collisions = new LinkedHashMap<>();
for (MavenProject project : projects) {
String projectId = ArtifactUtils.key(project.getGroupId(), project.getArtifactId(), project.getVersion());
MavenProject collision = index.get(projectId);
if (collision == null) {
index.put(projectId, project);
} else {
List pomFiles = collisions.get(projectId);
if (pomFiles == null) {
pomFiles = new ArrayList<>(Arrays.asList(collision.getFile(), project.getFile()));
collisions.put(projectId, pomFiles);
} else {
pomFiles.add(project.getFile());
}
}
}
if (!collisions.isEmpty()) {
throw new DuplicateProjectException(
"Two or more projects in the reactor"
+ " have the same identifier, please make sure that ::"
+ " is unique for each project: " + collisions,
collisions);
}
return index;
}
private Result extends ProjectDependencyGraph> buildGraph(MavenSession session, MavenExecutionResult result) {
Result extends ProjectDependencyGraph> graphResult = graphBuilder.build(session);
for (ModelProblem problem : graphResult.getProblems()) {
if (problem.getSeverity() == ModelProblem.Severity.WARNING) {
logger.warn(problem.toString());
} else {
logger.error(problem.toString());
}
}
if (!graphResult.hasErrors()) {
ProjectDependencyGraph projectDependencyGraph = graphResult.get();
session.setProjects(projectDependencyGraph.getSortedProjects());
session.setAllProjects(projectDependencyGraph.getAllProjects());
session.setProjectDependencyGraph(projectDependencyGraph);
}
return graphResult;
}
@Deprecated
// 5 January 2014
protected Logger getLogger() {
return logger;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy