org.netbeans.modules.maven.modelcache.MavenProjectCache Maven / Gradle / Ivy
The newest version!
/*
* 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.netbeans.modules.maven.modelcache;
import java.io.File;
import java.io.IOException;
import java.lang.ref.WeakReference;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.Collections;
import java.util.HashSet;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.Properties;
import java.util.Set;
import java.util.WeakHashMap;
import java.util.function.BiConsumer;
import java.util.logging.Level;
import java.util.logging.Logger;
import javax.swing.SwingUtilities;
import org.apache.maven.AbstractMavenLifecycleParticipant;
import org.apache.maven.artifact.Artifact;
import org.apache.maven.execution.DefaultMavenExecutionResult;
import org.apache.maven.execution.MavenExecutionRequest;
import org.apache.maven.execution.MavenExecutionResult;
import org.apache.maven.project.MavenProject;
import org.apache.maven.project.ProjectBuildingException;
import org.apache.maven.project.ProjectBuildingResult;
import org.codehaus.plexus.component.repository.exception.ComponentLookupException;
import org.netbeans.api.annotations.common.NonNull;
import org.netbeans.api.project.ProjectActionContext;
import org.netbeans.modules.maven.M2AuxilaryConfigImpl;
import org.netbeans.modules.maven.NbArtifactFixer;
import org.netbeans.modules.maven.api.NbMavenProject;
import org.netbeans.modules.maven.api.execute.RunConfig;
import org.netbeans.modules.maven.configurations.M2Configuration;
import org.netbeans.modules.maven.embedder.EmbedderFactory;
import org.netbeans.modules.maven.embedder.MavenEmbedder;
import org.netbeans.spi.project.AuxiliaryConfiguration;
import org.netbeans.spi.project.ProjectConfiguration;
import org.openide.filesystems.FileObject;
import org.openide.filesystems.FileUtil;
import org.openide.util.Mutex;
import org.openide.util.Mutex.Action;
import org.openide.util.NbBundle;
/**
* externalize the creation of MavenProject instances outside of NbMavenProjectImpl
* and be able to access it without a project at hand
* @author mkleint
*/
public final class MavenProjectCache {
private static final Logger LOG = Logger.getLogger(MavenProjectCache.class.getName());
private static final String CONTEXT_EXECUTION_RESULT = "NB_Execution_Result";
/**
* The value contains the set of placeholder artifacts, that have been injected during the reading operation.
*/
private static final String CONTEXT_PLACEHOLDER_ARTIFACTS = "NB_PlaceholderArtifacts";
/**
* Timestamp of the project model load.
*/
private static final String CONTEXT_LOAD_TIMESTAMP = "org.netbeans.modules.maven.loadTimestamp"; // NOI18N
/**
* Marks a project that observed unknown build participants.
*/
private static final String CONTEXT_PARTICIPANTS = "NB_AbstractParticipant_Present";
/**
* Marks a fallback project. The project was not fully read. Value of the context key is either a Boolean.TRUE as an indicator, or
* the partially read project, which does not resolve fully. Can be extracted with {@link #getPartialProject(org.apache.maven.project.MavenProject)}.
*/
private static final String CONTEXT_PARTIAL_PROJECT = "org.netbeans.modules.maven.partialProject"; // NOI18N
/**
* Marks a fallback project. The project is forged and is not read from the maven infrastructure. A fallback project may have a {@link #getPartialProject(org.apache.maven.project.MavenProject) partial project}
* attached.
*/
private static final String CONTEXT_FALLBACK_PROJECT = "org.netbeans.modules.maven.fallbackProject"; // NOI18N
/**
* Folder with module-configurable whitelist of lifecycle participants. Currently only 'ignore' can be specified.
*/
private static final String LIFECYCLE_PARTICIPANT_PREFIX = "Projects/" + NbMavenProject.TYPE + "/LifecycleParticipants/"; // NOI18N
/**
* Attribute that specifies the lifecycle participant should be silently ignored on model load.
*/
private static final String ATTR_IGNORE_ON_LOAD = "ignoreOnModelLoad"; // NOI18N
//File is referenced during lifetime of the Project. FileObject cannot be used as with rename it changes value@!!!
private static final Map> file2Project = new WeakHashMap>();
private static final Map file2Mutex = new WeakHashMap();
public static void clearMavenProject(final File pomFile) {
Mutex mutex = getMutex(pomFile);
mutex.writeAccess(new Action() {
@Override
public MavenProject run() {
file2Project.remove(pomFile);
return null;
}
});
}
/**
* returns a MavenProject instance for given folder, if folder contains a pom.xml always returns an instance, if not returns null
* @param pomFile
* @param reload consult the cache and return the cached value if true, otherwise force a reload.
* @return
*/
public static MavenProject getMavenProject(final File pomFile, final boolean reload) {
return getMavenProject(pomFile, false, reload);
}
public static MavenProject getMavenProject(final File pomFile, final boolean doNotLoadReturnNull, final boolean reload) {
Mutex mutex = getMutex(pomFile);
MavenProject mp = mutex.writeAccess(new Action() {
@Override
public MavenProject run() {
if (!reload) {
WeakReference ref = file2Project.get(pomFile);
if (ref != null) {
MavenProject mp = ref.get();
if (mp != null) {
LOG.log(Level.FINE, "Maven project {0} loaded from cache, packacing = {1}", new Object[] { pomFile, mp.getPackaging() });
return mp;
}
}
if (doNotLoadReturnNull) {
return null;
}
}
MavenProject mp = loadOriginalMavenProject(pomFile);
file2Project.put(pomFile, new WeakReference(mp));
return mp;
}
});
return mp;
}
public static MavenProject loadMavenProject(final File pomFile, ProjectActionContext context, RunConfig runConf) {
if (context == null) {
return getMavenProject(pomFile, true);
} else {
return loadOriginalMavenProject(pomFile, context, runConf);
}
}
public static MavenExecutionResult getExecutionResult(MavenProject project) {
return (MavenExecutionResult) project.getContextValue(CONTEXT_EXECUTION_RESULT);
}
public static boolean unknownBuildParticipantObserved(MavenProject project) {
return project.getContextValue(CONTEXT_PARTICIPANTS) != null;
}
/**
* Extracts a timestamp from the MavenProject. Timestamps are placed
* as context values in {@link #loadOriginalMavenProjectInternal}.
* @param p project
* @return timestamp, -1 if p is null, -2 if unknown
*/
public static long getLoadTimestamp(MavenProject p) {
if (p == null) {
return -1;
}
Object o = p.getContextValue(CONTEXT_LOAD_TIMESTAMP);
return o instanceof Long l ? l : -2;
}
/**
* Extracts the set of artifacts not present in the local repository and 'injected' during the
* project load by NbArtifactFixer.
* @param p project
* @return set of placeholder artifacts.
*/
public static Collection getPlaceholderArtifacts(MavenProject p) {
Object o = p.getContextValue(CONTEXT_PLACEHOLDER_ARTIFACTS);
return o instanceof Collection ? (Collection)o : Collections.emptySet();
}
/**
* list of class names of build participants in the project, null when none are present.
* @param project
* @return
*/
public static Collection getUnknownBuildParticipantsClassNames(MavenProject project) {
return (Collection) project.getContextValue(CONTEXT_PARTICIPANTS);
}
@NbBundle.Messages({
"TXT_RuntimeException=RuntimeException occurred in Apache Maven embedder while loading",
"TXT_RuntimeExceptionLong=RuntimeException occurred in Apache Maven embedder while loading the project. \n"
+ "This is preventing the project model from loading properly. \n"
+ "Please file a bug report with details about your project and the IDE's log file.\n\n"
})
private static @NonNull MavenProject loadOriginalMavenProject(final File pomFile) {
return loadOriginalMavenProject(pomFile, null, null);
}
private static boolean isLifecycleParticipatnIgnored(AbstractMavenLifecycleParticipant instance) {
String n = instance.getClass().getName();
FileObject check = FileUtil.getConfigFile(LIFECYCLE_PARTICIPANT_PREFIX + n);
return check != null && check.getAttribute(ATTR_IGNORE_ON_LOAD) == Boolean.TRUE;
}
public static MavenProject loadOriginalMavenProjectInternal(MavenEmbedder projectEmbedder, MavenExecutionRequest req) {
MavenProject newproject = null;
File pomFile = req.getPom();
MavenExecutionResult res = null;
Set placeholders = new HashSet<>();
long startLoading = System.currentTimeMillis();
try {
res = NbArtifactFixer.collectPlaceholderArtifacts(() -> projectEmbedder.readProjectWithDependencies(req, true), (c) ->
c.forEach(a -> {
// artifact fixer only injects POMs.
placeholders.add(projectEmbedder.createArtifactWithClassifier(a.getGroupId(), a.getArtifactId(), a.getVersion(), a.getExtension(), a.getClassifier())); // NOI18N
}
));
newproject = res.getProject();
//#204898
if (newproject != null) {
LOG.log(Level.FINE, "Loaded project for {0}, packaging: {1}", new Object[] { pomFile, newproject.getPackaging() });
ClassLoader projectRealm = newproject.getClassRealm();
if (projectRealm != null) {
ClassLoader originalClassLoader = Thread.currentThread().getContextClassLoader();
Thread.currentThread().setContextClassLoader(projectRealm);
try {
//boolean execute = EnableParticipantsBuildAction.isEnabled(aux);
List lookup = projectEmbedder.getPlexus().lookupList(AbstractMavenLifecycleParticipant.class);
if (lookup.size() > 0) { //just in case..
// if (execute) {
// LOG.info("Executing External Build Participants...");
// MavenSession session = new MavenSession( projectEmbedder.getPlexus(), newproject.getProjectBuildingRequest().getRepositorySession(), req, res );
// session.setCurrentProject(newproject);
// session.setProjects(Collections.singletonList(newproject));
// projectEmbedder.setUpLegacySupport();
// projectEmbedder.getPlexus().lookup(LegacySupport.class).setSession(session);
//
// for (AbstractMavenLifecycleParticipant part : lookup) {
// try {
// Thread.currentThread().setContextClassLoader( part.getClass().getClassLoader() );
// part.afterSessionStart(session);
// part.afterProjectsRead(session);
// } catch (MavenExecutionException ex) {
// Exceptions.printStackTrace(ex);
// }
// }
// } else {
List parts = new ArrayList();
for (AbstractMavenLifecycleParticipant part : lookup) {
if (isLifecycleParticipatnIgnored(part)) {
//#204898 create a whitelist of known not harmful participants that can be just ignored
continue;
}
String name = part.getClass().getName();
parts.add(name);
}
if (parts.size() > 0) {
newproject.setContextValue(CONTEXT_PARTICIPANTS, parts);
}
// }
}
} catch (ComponentLookupException e) {
// this is just silly, lookupList should return an empty list!
} finally {
Thread.currentThread().setContextClassLoader(originalClassLoader);
}
}
}
} catch (RuntimeException exc) {
//guard against exceptions that are not processed by the embedder
//#136184 NumberFormatException
LOG.log(Level.INFO, "Runtime exception thrown while loading maven project at " + pomFile, exc); //NOI18N
res = new DefaultMavenExecutionResult();
res.addException(exc);
} finally {
if (newproject == null) {
newproject = getFallbackProject(res, pomFile);
}
//#215159 clear the project building request, it references multiple Maven Models via the RepositorySession cache
//is not used in maven itself, most likely used by m2e only..
newproject.setProjectBuildingRequest(null);
long endLoading = System.currentTimeMillis();
LOG.log(Level.FINE, "Loaded project in {0} msec at {1}", new Object[] {endLoading - startLoading, pomFile.getPath()});
if (LOG.isLoggable(Level.FINE) && SwingUtilities.isEventDispatchThread()) {
LOG.log(Level.FINE, "Project " + pomFile.getPath() + " loaded in AWT event dispatching thread!", new RuntimeException());
}
if (LOG.isLoggable(Level.FINE) && !res.getExceptions().isEmpty()) {
LOG.log(Level.FINE, "Errors encountered during loading the project:");
for (Throwable t : res.getExceptions()) {
LOG.log(Level.FINE, "Maven reported:", t);
}
}
LOG.log(Level.FINE, "Loaded project flags - incomplete {0}, placeholder count {1}", new Object[] { isIncompleteProject(newproject), placeholders.size() });
if (!placeholders.isEmpty()) {
LOG.log(Level.FINE, "Incomplete artifact encountered during loading the project: {0}", placeholders);
if (!isIncompleteProject(newproject)) {
newproject.setContextValue(CONTEXT_PARTIAL_PROJECT, Boolean.TRUE);
}
newproject.setContextValue(CONTEXT_PLACEHOLDER_ARTIFACTS, placeholders);
}
}
newproject.setContextValue(CONTEXT_LOAD_TIMESTAMP, startLoading);
return newproject;
}
private static @NonNull MavenProject loadOriginalMavenProject(final File pomFile, ProjectActionContext ctx, RunConfig runConf) {
MavenEmbedder projectEmbedder = EmbedderFactory.getProjectEmbedder();
MavenProject newproject = null;
//TODO have independent from M2AuxiliaryConfigImpl
FileObject projectDir = FileUtil.toFileObject(pomFile.getParentFile());
if (projectDir == null || !projectDir.isValid()) {
LOG.log(Level.INFO, "Project directory is not valid: {0} from pom {1}, parent {2}", new Object[] { projectDir, pomFile, pomFile.getParentFile() });
return getFallbackProject(pomFile);
}
AuxiliaryConfiguration aux = new M2AuxilaryConfigImpl(projectDir, false);
ActiveConfigurationProvider config = new ActiveConfigurationProvider(projectDir, aux);
M2Configuration active;
active = config.getActiveConfiguration();
if (ctx != null && ctx.getConfiguration() != null) {
ProjectConfiguration cfg = ctx.getConfiguration();
if (cfg instanceof M2Configuration) {
active = (M2Configuration)cfg;
}
}
try {
List mavenConfigOpts = Collections.emptyList();
for (FileObject root = projectDir; root != null; root = root.getParent()) {
FileObject mavenConfig = root.getFileObject(".mvn/maven.config");
if (mavenConfig != null && mavenConfig.isData()) {
mavenConfigOpts = Arrays.asList(mavenConfig.asText().split("\\s+"));
LOG.log(Level.FINE, "Found maven config options: {0}", mavenConfigOpts);
break;
}
}
final MavenExecutionRequest req = projectEmbedder.createMavenExecutionRequest();
req.addActiveProfiles(active.getActivatedProfiles());
BiConsumer addActiveProfiles = (opt, prefix) -> req.addActiveProfiles(Arrays.asList(opt.substring(prefix.length()).split(",")));
Iterator optIt = mavenConfigOpts.iterator();
while (optIt.hasNext()) {
String opt = optIt.next();
// Could try to write/integrate a more general option parser here,
// but some options like -amd anyway violate GNU style.
if (opt.equals("-P") || opt.equals("--activate-profiles")) {
addActiveProfiles.accept(optIt.next(), "");
} else if (opt.startsWith("-P")) {
addActiveProfiles.accept(opt, "-P");
} else if (opt.startsWith("--activate-profiles=")) {
addActiveProfiles.accept(opt, "--activate-profiles=");
}
}
if (runConf != null) {
req.addActiveProfiles(runConf.getActivatedProfiles());
}
if (ctx != null && ctx.getProfiles() != null) {
req.addActiveProfiles(new ArrayList<>(ctx.getProfiles()));
}
req.setPom(pomFile);
req.setNoSnapshotUpdates(true);
req.setUpdateSnapshots(false);
//MEVENIDE-634 i'm wondering if this fixes the issue
req.setInteractiveMode(false);
// recursive == false is important to avoid checking all submodules for extensions
// that will not be used in current pom anyway..
// #135070
req.setRecursive(false);
req.setOffline(true);
//#238800 important to merge, not replace
Properties uprops = req.getUserProperties();
BiConsumer setProperty = (opt, prefix) -> {
String value = opt.substring(prefix.length());
int equals = value.indexOf('=');
if (equals != -1) {
uprops.setProperty(value.substring(0, equals), value.substring(equals + 1));
} else {
uprops.setProperty(value, "true");
}
};
optIt = mavenConfigOpts.iterator();
while (optIt.hasNext()) {
String opt = optIt.next();
if (opt.equals("-D") || opt.equals("--define")) {
setProperty.accept(optIt.next(), "");
} else if (opt.startsWith("-D")) {
setProperty.accept(opt, "-D");
} else if (opt.startsWith("--define=")) {
setProperty.accept(opt, "--define=");
}
}
uprops.putAll(createUserPropsForProjectLoading(active.getProperties()));
if (ctx != null && ctx.getProperties() != null) {
for (String k : ctx.getProperties().keySet()) {
uprops.setProperty(k, ctx.getProperties().get(k));
}
}
if (runConf != null && runConf.getProperties() != null) {
Map extends String, ? extends String> props = runConf.getProperties();
for (String k : props.keySet()) {
uprops.setProperty(k, props.get(k));
}
}
req.setUserProperties(uprops);
if (LOG.isLoggable(Level.FINE)) {
LOG.log(Level.FINE, "request property 'packaging': {0}", req.getSystemProperties().get("packaging"));
LOG.log(Level.FINE, "embedder property 'packaging': {0}", projectEmbedder.getSystemProperties().get("packaging"));
}
newproject = loadOriginalMavenProjectInternal(projectEmbedder, req);
} catch (IOException exc) {
MavenExecutionResult res = null;
//guard against exceptions that are not processed by the embedder
//#136184 NumberFormatException
LOG.log(Level.INFO, "Runtime exception thrown while loading maven project at " + pomFile, exc); //NOI18N
res = new DefaultMavenExecutionResult();
res.addException(exc);
newproject = getFallbackProject(res, pomFile);
}
return newproject;
}
/**
* Create a fallback project, but patch the incomplete project from the building result into it.
* The method will eventually start to return the partial project but still flagged as a fallback - see {@link #isFallbackproject(org.apache.maven.project.MavenProject)}.
*
* @param result the maven execution / project building result.
* @param projectFile the project file.
* @return fallback project
* @throws AssertionError
*/
public static MavenProject getFallbackProject(MavenExecutionResult result, File projectFile) throws AssertionError {
MavenProject toReturn = getFallbackProject(projectFile);
if (result == null) {
return toReturn;
}
toReturn.setContextValue(CONTEXT_EXECUTION_RESULT, result);
MavenProject partial = null;
for (Throwable t : result.getExceptions()) {
if (t instanceof ProjectBuildingException) {
ProjectBuildingException pbe = (ProjectBuildingException)t;
if (pbe.getResults() != null) {
for (ProjectBuildingResult res : pbe.getResults()) {
if (projectFile.equals(res.getPomFile())) {
partial = res.getProject();
break;
}
}
}
}
}
if (partial != null) {
toReturn.setContextValue(CONTEXT_PARTIAL_PROJECT, partial);
}
return toReturn;
}
@NbBundle.Messages({
"LBL_Incomplete_Project_Name=",
"LBL_Incomplete_Project_Desc=Partially loaded Maven project; try building it."
})
public static MavenProject getFallbackProject(File projectFile) throws AssertionError {
LOG.log(Level.FINE, "Creating fallback project for " + projectFile, new Throwable());
MavenProject newproject = new MavenProject();
newproject.setGroupId("error");
newproject.setArtifactId("error");
newproject.setVersion("0");
newproject.setPackaging("pom");
newproject.setName(Bundle.LBL_Incomplete_Project_Name());
newproject.setDescription(Bundle.LBL_Incomplete_Project_Desc());
newproject.setFile(projectFile);
newproject.setContextValue(CONTEXT_FALLBACK_PROJECT, true);
return newproject;
}
public static boolean isIncompleteProject(MavenProject prj) {
return prj.getContextValue(CONTEXT_PARTIAL_PROJECT) != null;
}
public static boolean isFallbackproject(MavenProject prj) {
if ("error".equals(prj.getGroupId()) && "error".equals(prj.getArtifactId()) && Bundle.LBL_Incomplete_Project_Name().equals(prj.getName())) {
return true;
} else {
return prj.getContextValue(CONTEXT_FALLBACK_PROJECT) != null;
}
}
public static MavenProject getPartialProject(MavenProject prj) {
Object o = prj.getContextValue(CONTEXT_PARTIAL_PROJECT);
if (o instanceof MavenProject) {
return (MavenProject)o;
} else {
return null;
}
}
public static Properties createUserPropsForProjectLoading(Map activeConfiguration) {
Properties props = new Properties();
if (activeConfiguration != null) {
props.putAll(activeConfiguration);
}
return props;
}
private static Mutex getMutex(File pomFile) {
synchronized (file2Mutex) {
Mutex mutex = file2Mutex.get(pomFile);
if (mutex != null) {
return mutex;
}
mutex = new Mutex();
file2Mutex.put(pomFile, mutex);
return mutex;
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy