From dd8c95c19305cf1c4aa29505164be439940df62c Mon Sep 17 00:00:00 2001 From: Tamas Cservenak Date: Sat, 8 Jun 2024 22:21:14 +0200 Subject: [PATCH] [MNG-8141][MNG-8147] Restore profile ID invariance but warn if duplicate IDs present (#1568) Fix and improvement in one PR as they are closely related. First, this PR restores the ability (broken by MNG-8081) to calculate Profile activation for POMs with duplicate Profile ID. Second, this PR improves UX by warning them about invalid models in their build. The reproducer now looks like this: https://gist.github.com/cstamas/165a610b233f4c03e381a0a2697903eb Notice: * WARNs issued about models (all Maven versions are mute about this) * still, property `${javafx.platform}` properly evaluated just like in 3.9.6 (but not in 3.9.7) * build succeeds (fails in 3.9.7) --- https://issues.apache.org/jira/browse/MNG-8147 https://issues.apache.org/jira/browse/MNG-8141 --- .../model/building/DefaultModelBuilder.java | 69 +++++++------------ .../DefaultArtifactDescriptorReader.java | 28 +++++++- 2 files changed, 53 insertions(+), 44 deletions(-) diff --git a/maven-model-builder/src/main/java/org/apache/maven/model/building/DefaultModelBuilder.java b/maven-model-builder/src/main/java/org/apache/maven/model/building/DefaultModelBuilder.java index 3043a76a0ea6..26ace56496c0 100644 --- a/maven-model-builder/src/main/java/org/apache/maven/model/building/DefaultModelBuilder.java +++ b/maven-model-builder/src/main/java/org/apache/maven/model/building/DefaultModelBuilder.java @@ -28,6 +28,7 @@ import java.util.Collection; import java.util.Collections; import java.util.HashMap; +import java.util.HashSet; import java.util.Iterator; import java.util.LinkedHashSet; import java.util.List; @@ -35,9 +36,7 @@ import java.util.Objects; import java.util.Optional; import java.util.Properties; -import java.util.Set; import java.util.function.Consumer; -import java.util.stream.Collectors; import org.apache.maven.artifact.versioning.DefaultArtifactVersion; import org.apache.maven.artifact.versioning.InvalidVersionSpecificationException; @@ -305,18 +304,17 @@ protected ModelBuildingResult build(ModelBuildingRequest request, Collection interpolatedActivations = - getInterpolatedActivations(rawModel, profileActivationContext, problems); - injectProfileActivations(tmpModel, interpolatedActivations); + List interpolatedProfiles = getInterpolatedProfiles(rawModel, profileActivationContext, problems); + tmpModel.setProfiles(interpolatedProfiles); List activePomProfiles = profileSelector.getActiveProfiles(tmpModel.getProfiles(), profileActivationContext, problems); - Set activeProfileIds = - activePomProfiles.stream().map(Profile::getId).collect(Collectors.toSet()); - currentData.setActiveProfiles(rawModel.getProfiles().stream() - .filter(p -> activeProfileIds.contains(p.getId())) - .collect(Collectors.toList())); + List rawProfiles = new ArrayList<>(); + for (Profile activePomProfile : activePomProfiles) { + rawProfiles.add(rawModel.getProfiles().get(interpolatedProfiles.indexOf(activePomProfile))); + } + currentData.setActiveProfiles(rawProfiles); // profile injection for (Profile activeProfile : activePomProfiles) { @@ -429,12 +427,12 @@ private interface InterpolateString { String apply(String s) throws InterpolationException; } - private Map getInterpolatedActivations( + private List getInterpolatedProfiles( Model rawModel, DefaultProfileActivationContext context, DefaultModelProblemCollector problems) { - Map interpolatedActivations = getProfileActivations(rawModel, true); + List interpolatedActivations = getProfiles(rawModel, true); if (interpolatedActivations.isEmpty()) { - return Collections.emptyMap(); + return Collections.emptyList(); } RegexBasedInterpolator interpolator = new RegexBasedInterpolator(); @@ -466,8 +464,14 @@ void performFor(String value, String locationKey, Consumer mutator) { } } } - for (Activation activation : interpolatedActivations.values()) { - Optional a = Optional.of(activation); + HashSet profileIds = new HashSet<>(); + for (Profile profile : interpolatedActivations) { + if (!profileIds.add(profile.getId())) { + problems.add(new ModelProblemCollectorRequest(Severity.WARNING, ModelProblem.Version.BASE) + .setMessage("Duplicate activation for profile " + profile.getId())); + } + Activation activation = profile.getActivation(); + Optional a = Optional.ofNullable(activation); a.map(Activation::getFile).ifPresent(fa -> { Interpolation nt = new Interpolation(fa, s -> profileActivationFilePathInterpolator.interpolate(s, context)); @@ -753,41 +757,20 @@ private void assembleInheritance( } } - private Map getProfileActivations(Model model, boolean clone) { - Map activations = new HashMap<>(); + private List getProfiles(Model model, boolean clone) { + ArrayList profiles = new ArrayList<>(); for (Profile profile : model.getProfiles()) { - Activation activation = profile.getActivation(); - - if (activation == null) { - continue; - } - if (clone) { - activation = activation.clone(); + profile = profile.clone(); } - - activations.put(profile.getId(), activation); - } - - return activations; - } - - private void injectProfileActivations(Model model, Map activations) { - for (Profile profile : model.getProfiles()) { - Activation activation = profile.getActivation(); - - if (activation == null) { - continue; - } - - // restore activation - profile.setActivation(activations.get(profile.getId())); + profiles.add(profile); } + return profiles; } private Model interpolateModel(Model model, ModelBuildingRequest request, ModelProblemCollector problems) { // save profile activations before interpolation, since they are evaluated with limited scope - Map originalActivations = getProfileActivations(model, true); + List originalActivations = getProfiles(model, true); Model interpolatedModel = modelInterpolator.interpolateModel(model, model.getProjectDirectory(), request, problems); @@ -815,7 +798,7 @@ private Model interpolateModel(Model model, ModelBuildingRequest request, ModelP interpolatedModel.setPomFile(model.getPomFile()); // restore profiles with file activation to their value before full interpolation - injectProfileActivations(model, originalActivations); + model.setProfiles(originalActivations); return interpolatedModel; } diff --git a/maven-resolver-provider/src/main/java/org/apache/maven/repository/internal/DefaultArtifactDescriptorReader.java b/maven-resolver-provider/src/main/java/org/apache/maven/repository/internal/DefaultArtifactDescriptorReader.java index bbbc463e86c7..4108d642f98f 100644 --- a/maven-resolver-provider/src/main/java/org/apache/maven/repository/internal/DefaultArtifactDescriptorReader.java +++ b/maven-resolver-provider/src/main/java/org/apache/maven/repository/internal/DefaultArtifactDescriptorReader.java @@ -23,6 +23,7 @@ import javax.inject.Singleton; import java.util.LinkedHashSet; +import java.util.List; import java.util.Map; import java.util.Objects; import java.util.Properties; @@ -37,7 +38,9 @@ import org.apache.maven.model.building.ModelBuilder; import org.apache.maven.model.building.ModelBuildingException; import org.apache.maven.model.building.ModelBuildingRequest; +import org.apache.maven.model.building.ModelBuildingResult; import org.apache.maven.model.building.ModelProblem; +import org.apache.maven.model.building.ModelProblemUtils; import org.apache.maven.model.resolution.UnresolvableModelException; import org.eclipse.aether.RepositoryEvent; import org.eclipse.aether.RepositoryEvent.EventType; @@ -67,6 +70,8 @@ import org.eclipse.aether.spi.locator.Service; import org.eclipse.aether.spi.locator.ServiceLocator; import org.eclipse.aether.transfer.ArtifactNotFoundException; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; /** * @author Benjamin Bentmann @@ -91,6 +96,8 @@ public class DefaultArtifactDescriptorReader implements ArtifactDescriptorReader private final ArtifactDescriptorReaderDelegate artifactDescriptorReaderDelegate = new ArtifactDescriptorReaderDelegate(); + private final Logger logger = LoggerFactory.getLogger(getClass()); + @Deprecated public DefaultArtifactDescriptorReader() { // enable no-arg constructor @@ -281,7 +288,26 @@ private Model loadPom( modelRequest.setModelSource(new FileModelSource(pomArtifact.getFile())); } - model = modelBuilder.build(modelRequest).getEffectiveModel(); + ModelBuildingResult modelResult = modelBuilder.build(modelRequest); + // ModelBuildingEx is thrown only on FATAL and ERROR severities, but we still can have WARNs + // that may lead to unexpected build failure, log them + if (!modelResult.getProblems().isEmpty()) { + List problems = modelResult.getProblems(); + logger.warn( + "{} {} encountered while building the effective model for {}", + problems.size(), + (problems.size() == 1) ? "problem was" : "problems were", + request.getArtifact()); + if (logger.isDebugEnabled()) { + for (ModelProblem problem : problems) { + logger.warn( + "{} @ {}", + problem.getMessage(), + ModelProblemUtils.formatLocation(problem, null)); + } + } + } + model = modelResult.getEffectiveModel(); } catch (ModelBuildingException e) { for (ModelProblem problem : e.getProblems()) { if (problem.getException() instanceof UnresolvableModelException) {