org.openrewrite.maven.tree.ResolvedPom Maven / Gradle / Ivy
/*
* Copyright 2021 the original author or authors.
*
* Licensed 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
*
* https://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.openrewrite.maven.tree;
import com.fasterxml.jackson.annotation.JsonCreator;
import com.fasterxml.jackson.annotation.JsonIdentityInfo;
import com.fasterxml.jackson.annotation.ObjectIdGenerators;
import com.fasterxml.jackson.databind.JsonNode;
import com.fasterxml.jackson.databind.node.ArrayNode;
import com.fasterxml.jackson.databind.node.JsonNodeFactory;
import com.fasterxml.jackson.databind.node.ObjectNode;
import lombok.Builder;
import lombok.Getter;
import lombok.Value;
import lombok.With;
import lombok.experimental.NonFinal;
import org.jspecify.annotations.Nullable;
import org.openrewrite.ExecutionContext;
import org.openrewrite.internal.ListUtils;
import org.openrewrite.internal.PropertyPlaceholderHelper;
import org.openrewrite.maven.MavenDownloadingException;
import org.openrewrite.maven.MavenDownloadingExceptions;
import org.openrewrite.maven.MavenExecutionContextView;
import org.openrewrite.maven.cache.MavenPomCache;
import org.openrewrite.maven.internal.MavenParsingException;
import org.openrewrite.maven.internal.MavenPomDownloader;
import org.openrewrite.maven.internal.VersionRequirement;
import org.openrewrite.maven.tree.ManagedDependency.Defined;
import org.openrewrite.maven.tree.ManagedDependency.Imported;
import org.openrewrite.maven.tree.Plugin.Execution;
import java.util.*;
import java.util.function.Function;
import java.util.function.UnaryOperator;
import java.util.stream.Collectors;
import static java.util.Collections.*;
import static org.openrewrite.internal.StringUtils.matchesGlob;
@JsonIdentityInfo(generator = ObjectIdGenerators.IntSequenceGenerator.class, property = "@ref")
@Getter
@Builder
public class ResolvedPom {
public static final PropertyPlaceholderHelper placeholderHelper = new PropertyPlaceholderHelper("${", "}", null);
// https://maven.apache.org/ref/3.6.3/maven-model-builder/super-pom.html
private static final ResolvedPom SUPER_POM = ResolvedPom.builder()
.repositories(singletonList(MavenRepository.MAVEN_CENTRAL))
.build();
@With
Pom requested;
@With
Iterable activeProfiles;
public ResolvedPom(Pom requested, Iterable activeProfiles) {
this(requested, activeProfiles, emptyMap(), emptyList(), null, emptyList(), emptyList(), emptyList(), emptyList(), emptyList());
}
@JsonCreator
ResolvedPom(Pom requested, Iterable activeProfiles, Map properties, List dependencyManagement, @Nullable List initialRepositories, List repositories, List requestedDependencies, List plugins, List pluginManagement, List subprojects) {
this.requested = requested;
this.activeProfiles = activeProfiles;
this.properties = properties;
this.dependencyManagement = dependencyManagement;
this.initialRepositories = initialRepositories;
this.repositories = repositories;
this.requestedDependencies = requestedDependencies;
this.plugins = plugins;
this.pluginManagement = pluginManagement;
this.subprojects = subprojects;
}
@NonFinal
@Builder.Default
Map properties = emptyMap();
@NonFinal
@Builder.Default
List dependencyManagement = emptyList();
@NonFinal
@Builder.Default
List initialRepositories = emptyList();
@NonFinal
@Builder.Default
List repositories = emptyList();
@NonFinal
@Builder.Default
List requestedDependencies = emptyList();
@NonFinal
@Builder.Default
List plugins = emptyList();
@NonFinal
@Builder.Default
List pluginManagement = emptyList();
@NonFinal
@Builder.Default
@Nullable // on older LSTs, this field is not yet present
List subprojects = emptyList();
/**
* Deduplicate dependencies and dependency management dependencies
*
* @return This POM after deduplication.
*/
@SuppressWarnings("UnnecessaryLocalVariable")
public ResolvedPom deduplicate() {
Set uniqueManagedDependencies = new HashSet<>(dependencyManagement.size());
List dedupMd = ListUtils.map(dependencyManagement, dm -> uniqueManagedDependencies.add(new UniqueDependencyKey(dm.getGav(), dm.getType(), dm.getClassifier(), dm.getScope())) ?
dm : null);
dependencyManagement = dedupMd;
uniqueManagedDependencies.clear();
List dedupD = ListUtils.map(requestedDependencies, d -> uniqueManagedDependencies.add(new UniqueDependencyKey(d.getGav(), d.getType(), d.getClassifier(), d.getScope())) ?
d : null);
requestedDependencies = dedupD;
return this;
}
@Value
private static class UniqueDependencyKey {
GroupArtifactVersion gav;
@Nullable
String type;
@Nullable
String classifier;
Object scope;
}
/**
* Whenever a change is made that may affect the effective properties, dependency management,
* dependencies, etc. of a POM, this can be called to re-resolve the POM.
*
* @param ctx An execution context containing any maven-specific requirements.
* @param downloader A POM downloader to download dependencies and parents.
* @return A new instance with dependencies re-resolved or the same instance if no resolved dependencies have changed.
* @throws MavenDownloadingException When problems are encountered downloading dependencies or parents.
*/
@SuppressWarnings("DuplicatedCode")
public ResolvedPom resolve(ExecutionContext ctx, MavenPomDownloader downloader) throws MavenDownloadingException {
// If this resolved pom represents an obsolete pom format, refuse to resolve in same as Maven itself would
if (requested.getObsoletePomVersion() != null) {
return this;
}
ResolvedPom resolved = new ResolvedPom(
requested,
activeProfiles,
emptyMap(),
emptyList(),
initialRepositories,
emptyList(),
emptyList(),
emptyList(),
emptyList(),
emptyList()
).resolver(ctx, downloader).resolve();
for (Map.Entry property : resolved.getProperties().entrySet()) {
if (properties == null || (property.getValue() != null && !property.getValue().equals(properties.get(property.getKey())))) {
return resolved;
}
}
List resolvedRequestedDependencies = resolved.getRequestedDependencies();
if (requestedDependencies == null || requestedDependencies.size() != resolvedRequestedDependencies.size()) {
return resolved;
}
for (int i = 0; i < resolvedRequestedDependencies.size(); i++) {
if (!requestedDependencies.get(i).equals(resolvedRequestedDependencies.get(i))) {
return resolved;
}
}
List resolvedDependencyManagement = resolved.getDependencyManagement();
if (dependencyManagement == null || dependencyManagement.size() != resolvedDependencyManagement.size()) {
return resolved;
}
for (int i = 0; i < resolvedDependencyManagement.size(); i++) {
// TODO does ResolvedPom's equals work well enough to match on BOM imports?
if (!dependencyManagement.get(i).equals(resolvedDependencyManagement.get(i))) {
return resolved;
}
}
List resolvedRepositories = resolved.getRepositories();
if (repositories == null || repositories.size() != resolvedRepositories.size()) {
return resolved;
}
for (int i = 0; i < resolvedRepositories.size(); i++) {
if (!repositories.get(i).equals(resolvedRepositories.get(i))) {
return resolved;
}
}
List resolvedPlugins = resolved.getPlugins();
if (plugins == null || plugins.size() != resolvedPlugins.size()) {
return resolved;
}
for (int i = 0; i < resolvedPlugins.size(); i++) {
if (!plugins.get(i).equals(resolvedPlugins.get(i))) {
return resolved;
}
}
List resolvedPluginManagement = resolved.getPluginManagement();
if (pluginManagement == null || pluginManagement.size() != resolvedPluginManagement.size()) {
return resolved;
}
for (int i = 0; i < resolvedPluginManagement.size(); i++) {
if (!pluginManagement.get(i).equals(resolvedPluginManagement.get(i))) {
return resolved;
}
}
return this;
}
Resolver resolver(ExecutionContext ctx, MavenPomDownloader downloader) {
return new Resolver(ctx, downloader);
}
public ResolvedGroupArtifactVersion getGav() {
return requested.getGav();
}
public String getGroupId() {
return requested.getGroupId();
}
public String getArtifactId() {
return requested.getArtifactId();
}
public String getVersion() {
return requested.getVersion();
}
@SuppressWarnings("unused")
public @Nullable String getDatedSnapshotVersion() {
return requested.getDatedSnapshotVersion();
}
public String getPackaging() {
return requested.getPackaging() == null ? "jar" : requested.getPackaging();
}
public @Nullable String getValue(@Nullable String value) {
if (value == null) {
return null;
}
return placeholderHelper.replacePlaceholders(value, this::getProperty);
}
private @Nullable String getProperty(@Nullable String property) {
if (property == null) {
return null;
}
String propVal = properties.get(property);
if (propVal != null) {
return propVal;
}
switch (property) {
case "groupId":
case "project.groupId":
case "pom.groupId":
return requested.getGroupId();
case "project.parent.groupId":
case "parent.groupId":
return requested.getParent() != null ? requested.getParent().getGroupId() : null;
case "artifactId":
case "project.artifactId":
case "pom.artifactId":
return requested.getArtifactId(); // cannot be inherited from parent
case "project.parent.artifactId":
case "parent.artifactId":
return requested.getParent() == null ? null : requested.getParent().getArtifactId();
case "version":
case "project.version":
case "pom.version":
return requested.getVersion();
case "project.parent.version":
case "parent.version":
return requested.getParent() != null ? requested.getParent().getVersion() : null;
case "prerequisites.maven":
case "pom.prerequisites.maven":
case "project.prerequisites.maven":
return requested.getPrerequisites() == null ? null : requested.getPrerequisites().getMaven();
}
return System.getProperty(property);
}
public @Nullable String getManagedVersion(@Nullable String groupId, String artifactId, @Nullable String type, @Nullable String classifier) {
for (ResolvedManagedDependency dm : dependencyManagement) {
if (dm.matches(groupId, artifactId, type, classifier)) {
return getValue(dm.getVersion());
}
}
return null;
}
public List getManagedExclusions(String groupId, String artifactId, @Nullable String type, @Nullable String classifier) {
for (ResolvedManagedDependency dm : dependencyManagement) {
if (dm.matches(groupId, artifactId, type, classifier)) {
return dm.getExclusions() == null ? emptyList() : dm.getExclusions();
}
}
return emptyList();
}
public @Nullable Scope getManagedScope(String groupId, String artifactId, @Nullable String type, @Nullable String classifier) {
for (ResolvedManagedDependency dm : dependencyManagement) {
if (dm.matches(groupId, artifactId, type, classifier)) {
return dm.getScope();
}
}
return null;
}
public GroupArtifactVersion getValues(GroupArtifactVersion gav) {
return gav.withGroupId(getValue(gav.getGroupId()))
.withArtifactId(getValue(gav.getArtifactId()))
.withVersion(getValue(gav.getVersion()));
}
public GroupArtifact getValues(GroupArtifact ga) {
return ga.withGroupId(getValue(ga.getGroupId()))
.withArtifactId(getValue(ga.getArtifactId()));
}
@Value
class Resolver {
ExecutionContext ctx;
MavenPomDownloader downloader;
public ResolvedPom resolve() throws MavenDownloadingException {
resolveParentsRecursively(requested);
return ResolvedPom.this;
}
void resolveParentsRecursively(Pom requested) throws MavenDownloadingException {
List pomAncestry = new ArrayList<>();
pomAncestry.add(requested);
if (initialRepositories != null) {
mergeRepositories(initialRepositories);
}
resolveParentPropertiesAndRepositoriesRecursively(new ArrayList<>(pomAncestry));
if (initialRepositories == null) {
initialRepositories = repositories;
}
//Once properties have been merged, update any property placeholders in the resolved gav
//coordinates. This is important to do early because any system properties used within the coordinates
//are transient and will not be available once pom has been serialized/deserialized into a different VM.
Pom pomReference = ResolvedPom.this.requested;
pomReference = pomReference.withGav(pomReference.getGav().withRepository(getValue(pomReference.getGav().getRepository())));
pomReference = pomReference.withGav(pomReference.getGav().withGroupId(getValue(pomReference.getGav().getGroupId())));
pomReference = pomReference.withGav(pomReference.getGav().withArtifactId(getValue(pomReference.getGav().getArtifactId())));
pomReference = pomReference.withGav(pomReference.getGav().withVersion(getValue(pomReference.getGav().getVersion())));
pomReference = pomReference.withGav(pomReference.getGav().withDatedSnapshotVersion(getValue(pomReference.getGav().getDatedSnapshotVersion())));
if (ResolvedPom.this.requested != pomReference) {
ResolvedPom.this.requested = pomReference;
}
resolveParentDependenciesRecursively(new ArrayList<>(pomAncestry));
resolveParentPluginsRecursively(new ArrayList<>(pomAncestry));
}
private void resolveParentPropertiesAndRepositoriesRecursively(List pomAncestry) throws MavenDownloadingException {
Pom pom = pomAncestry.get(0);
//Resolve properties
for (Profile profile : pom.getProfiles()) {
if (profile.isActive(activeProfiles)) {
mergeProperties(profile.getProperties(), pom);
}
}
mergeProperties(pom.getProperties(), pom);
//Resolve repositories (which may rely on properties ^^^)
for (Profile profile : pom.getProfiles()) {
if (profile.isActive(activeProfiles)) {
mergeRepositories(profile.getRepositories());
}
}
mergeRepositories(pom.getRepositories());
if (pom.getParent() != null) {
Pom parentPom = resolveParentPom(pom);
for (Pom ancestor : pomAncestry) {
if (ancestor.getGav().equals(parentPom.getGav())) {
// parent cycle
return;
}
}
pomAncestry.add(0, parentPom);
resolveParentPropertiesAndRepositoriesRecursively(pomAncestry);
}
}
private void resolveParentDependenciesRecursively(List pomAncestry) throws MavenDownloadingException {
Pom pom = pomAncestry.get(0);
for (Profile profile : pom.getProfiles()) {
if (profile.isActive(activeProfiles)) {
mergeDependencyManagement(profile.getDependencyManagement(), pomAncestry);
mergeRequestedDependencies(profile.getDependencies());
}
}
mergeDependencyManagement(pom.getDependencyManagement(), pomAncestry);
mergeRequestedDependencies(pom.getDependencies());
if (pom.getParent() != null) {
Pom parentPom = resolveParentPom(pom);
MavenExecutionContextView.view(ctx)
.getResolutionListener()
.parent(parentPom, pom);
for (Pom ancestor : pomAncestry) {
if (ancestor.getGav().equals(parentPom.getGav())) {
// parent cycle
return;
}
}
pomAncestry.add(0, parentPom);
resolveParentDependenciesRecursively(pomAncestry);
}
}
private void resolveParentPluginsRecursively(List pomAncestry) throws MavenDownloadingException {
Pom pom = pomAncestry.get(0);
for (Profile profile : pom.getProfiles()) {
if (profile.isActive(activeProfiles)) {
mergePluginManagement(profile.getPluginManagement());
mergePlugins(profile.getPlugins());
}
}
mergePluginManagement(pom.getPluginManagement());
mergePlugins(pom.getPlugins());
if (pom.getParent() != null) {
Pom parentPom = resolveParentPom(pom);
MavenExecutionContextView.view(ctx)
.getResolutionListener()
.parent(parentPom, pom);
for (Pom ancestor : pomAncestry) {
if (ancestor.getGav().equals(parentPom.getGav())) {
// parent cycle
return;
}
}
pomAncestry.add(0, parentPom);
resolveParentPluginsRecursively(pomAncestry);
}
}
private Pom resolveParentPom(Pom pom) throws MavenDownloadingException {
@SuppressWarnings("DataFlowIssue") GroupArtifactVersion gav = getValues(pom.getParent().getGav());
if (gav.getVersion() == null) {
throw new MavenParsingException("Parent version must always specify a version " + gav);
}
VersionRequirement newRequirement = VersionRequirement.fromVersion(gav.getVersion(), 0);
GroupArtifact ga = new GroupArtifact(gav.getGroupId(), gav.getArtifactId());
String newRequiredVersion = newRequirement.resolve(ga, downloader, getRepositories());
if (newRequiredVersion == null) {
throw new MavenParsingException("Could not resolve version for [" + ga + "] matching version requirements " + newRequirement);
}
gav = gav.withVersion(newRequiredVersion);
return downloader.download(gav,
pom.getParent().getRelativePath(), ResolvedPom.this, repositories);
}
private void mergeRequestedDependencies(List incomingRequestedDependencies) {
if (!incomingRequestedDependencies.isEmpty()) {
if (requestedDependencies == null || requestedDependencies.isEmpty()) {
//It is possible for the dependencies to be an empty, immutable list.
//If it's empty, we ensure to create a mutable list.
requestedDependencies = new ArrayList<>(incomingRequestedDependencies);
} else {
// When a child dependency has overriden a parent dependency (either version or scope)
// We shouldn't add the parent definition when requested; the child takes precedence
for (Dependency incReqDep : incomingRequestedDependencies) {
boolean found = false;
for (Dependency reqDep : requestedDependencies) {
if (reqDep.getGav().getGroupId().equals(incReqDep.getGav().getGroupId()) &&
reqDep.getArtifactId().equals(incReqDep.getArtifactId())) {
found = true;
break;
}
}
if (!found) {
requestedDependencies.add(incReqDep);
}
}
}
}
}
@Value
private class PluginKey {
String groupId;
String artifactId;
}
private PluginKey getPluginKey(Plugin plugin) {
return new PluginKey(
plugin.getGroupId(),
plugin.getArtifactId()
);
}
private List mergePluginDependencies(List dependencies, List incomingDependencies) {
if (incomingDependencies.isEmpty()) {
return dependencies;
}
if (dependencies.isEmpty()) {
return incomingDependencies;
}
List merged = new ArrayList<>();
Set uniqueDependencies = new HashSet<>();
for (Dependency dependency : dependencies) {
merged.add(dependency);
uniqueDependencies.add(new GroupArtifact(dependency.getGroupId(), dependency.getArtifactId()));
}
for (Dependency dependency : incomingDependencies) {
if (!uniqueDependencies.contains(new GroupArtifact(dependency.getGroupId(), dependency.getArtifactId()))) {
merged.add(dependency);
}
}
return merged;
}
private @Nullable JsonNode mergePluginConfigurations(@Nullable JsonNode configuration, @Nullable JsonNode incomingConfiguration) {
if (!(incomingConfiguration instanceof ObjectNode)) {
return configuration;
}
if (!(configuration instanceof ObjectNode)) {
return incomingConfiguration;
}
ObjectNode ret = incomingConfiguration.deepCopy();
Iterator> fields = configuration.fields();
while (fields.hasNext()) {
Map.Entry conf = fields.next();
JsonNode parentConf = ret.get(conf.getKey());
JsonNode parentCombine = parentConf != null ? parentConf.get("combine.children") : null;
if (parentCombine != null && "append".equals(parentCombine.asText())) {
JsonNode selfCombine = conf.getValue().get("combine.self");
if (selfCombine != null && "override".equals(selfCombine.asText())) {
ret.set(conf.getKey(), conf.getValue());
} else {
ret.set(conf.getKey(), combineLists(conf.getValue(), parentConf));
}
} else {
ret.set(conf.getKey(), conf.getValue());
}
}
return ret;
}
private JsonNode combineLists(JsonNode list, JsonNode incomingList) {
ObjectNode ret = incomingList.deepCopy();
ArrayList keys = new ArrayList<>();
ret.fieldNames().forEachRemaining(keys::add);
keys.remove("combine.children");
// If no keys remaining, it's an empty list, we return the other one
if (keys.isEmpty()) {
return list.deepCopy();
}
// We can only have one key remaining in a list
String arrayElemField = keys.get(0);
// Copy elements of the list
JsonNode retNode = ret.get(arrayElemField);
JsonNode node = list.get(arrayElemField);
if (!(retNode instanceof ArrayNode)) {
ArrayNode arrayNode = JsonNodeFactory.instance.arrayNode();
arrayNode.add(retNode);
ret.set(arrayElemField, arrayNode);
retNode = arrayNode;
}
if (node instanceof ArrayNode) {
((ArrayNode) retNode).addAll((ArrayNode) node);
} else if (node != null) {
((ArrayNode) retNode).add(node);
}
// Check if combine.children is overridden
JsonNode listCombine = list.get("combine.children");
if (listCombine != null) {
ret.set("combine.children", listCombine);
}
return ret;
}
private List mergePluginExecutions(List currentExecutions, List incomingExecutions) {
if (currentExecutions.isEmpty()) {
return incomingExecutions;
}
if (incomingExecutions.isEmpty()) {
return currentExecutions;
}
Map currentExecutionsById = currentExecutions.stream()
.collect(Collectors.toMap(Execution::getId, Function.identity()));
List mergedExecutions = new ArrayList<>(currentExecutions);
for (Plugin.Execution incomingExecution : incomingExecutions) {
String executionId = incomingExecution.getId();
if (!currentExecutionsById.containsKey(executionId)) {
mergedExecutions.add(incomingExecution);
} else {
Plugin.Execution currentExecution = currentExecutionsById.get(executionId);
// GOALS
Set mergedGoals = new HashSet<>();
if (currentExecution.getGoals() != null) {
mergedGoals.addAll(currentExecution.getGoals());
}
if (incomingExecution.getGoals() != null) {
mergedGoals.addAll(incomingExecution.getGoals());
}
// PHASE
String mergedPhase = currentExecution.getPhase();
if (incomingExecution.getPhase() != null &&
!Objects.equals(mergedPhase, incomingExecution.getPhase())) {
mergedPhase = incomingExecution.getPhase();
}
// CONFIGURATION
JsonNode mergedConfiguration = mergePluginConfigurations(
currentExecution.getConfiguration(),
incomingExecution.getConfiguration());
// EXECUTION
Plugin.Execution mergedExecution = new Plugin.Execution(
executionId,
new ArrayList<>(mergedGoals),
mergedPhase,
incomingExecution.getInherited(),
mergedConfiguration);
mergedExecutions.remove(currentExecution);
mergedExecutions.add(mergedExecution);
}
}
return mergedExecutions;
}
private Plugin mergePlugins(Plugin plugin, Plugin incoming) {
return new Plugin(
plugin.getGroupId(),
plugin.getArtifactId(),
Optional.ofNullable(plugin.getVersion()).orElse(incoming.getVersion()),
Optional.ofNullable(plugin.getExtensions()).orElse(incoming.getExtensions()),
Optional.ofNullable(plugin.getInherited()).orElse(incoming.getInherited()),
mergePluginConfigurations(plugin.getConfiguration(), incoming.getConfiguration()),
mergePluginDependencies(plugin.getDependencies(), incoming.getDependencies()),
mergePluginExecutions(plugin.getExecutions(), incoming.getExecutions())
);
}
private void mergePlugins(List plugins, List incomingPlugins) {
Map pluginMap = new HashMap<>();
plugins.forEach(p -> pluginMap.put(getPluginKey(p), p));
for (Plugin incomingPlugin : incomingPlugins) {
if ("false".equals(incomingPlugin.getInherited())) {
continue;
}
Plugin plugin = pluginMap.get(getPluginKey(incomingPlugin));
if (plugin != null) {
plugins.remove(plugin);
plugins.add(mergePlugins(plugin, incomingPlugin));
} else {
plugins.add(incomingPlugin);
}
}
}
private void mergePlugins(List incomingPlugins) {
if (!incomingPlugins.isEmpty()) {
if (plugins == null || plugins.isEmpty()) {
//It is possible for the plugins to be an empty, immutable list.
//If it's empty, we ensure to create a mutable list.
plugins = new ArrayList<>();
}
mergePlugins(plugins, incomingPlugins);
}
}
private void mergePluginManagement(List incomingPlugins) {
if (!incomingPlugins.isEmpty()) {
if (pluginManagement == null || pluginManagement.isEmpty()) {
//It is possible for the plugins to be an empty, immutable list.
//If it's empty, we ensure to create a mutable list.
pluginManagement = new ArrayList<>();
}
mergePlugins(pluginManagement, incomingPlugins);
}
}
private void mergeRepositories(List incomingRepositories) {
if (!incomingRepositories.isEmpty()) {
if (repositories == null || repositories.isEmpty()) {
//It is possible for the repositories to be an empty, immutable list.
//If it's empty, we ensure to create a mutable list.
repositories = new ArrayList<>(incomingRepositories.size());
}
nextRepository:
for (MavenRepository incomingRepository : incomingRepositories) {
@SuppressWarnings("ConstantConditions")
MavenRepository incoming = new MavenRepository(
getValue(incomingRepository.getId()),
getValue(incomingRepository.getUri()),
incomingRepository.getReleases(),
incomingRepository.getSnapshots(),
incomingRepository.isKnownToExist(),
incomingRepository.getUsername(),
incomingRepository.getPassword(),
incomingRepository.getTimeout(),
incomingRepository.getDeriveMetadataIfMissing()
);
if (incoming.getId() != null) {
for (MavenRepository repository : repositories) {
if (incoming.getId().equals(repository.getId())) {
continue nextRepository;
}
}
}
repositories.add(incoming);
}
}
}
private void mergeProperties(Map incomingProperties, Pom pom) {
if (!incomingProperties.isEmpty()) {
if (properties == null || properties.isEmpty()) {
//It is possible for the properties to be an empty, immutable map.
//If it's empty, we ensure to create a mutable map.
properties = new HashMap<>(incomingProperties.size());
}
for (Map.Entry property : incomingProperties.entrySet()) {
MavenExecutionContextView.view(ctx)
.getResolutionListener()
.property(property.getKey(), property.getValue(), pom);
if (!properties.containsKey(property.getKey())) {
properties.put(property.getKey(), property.getValue());
}
}
}
}
private void mergeDependencyManagement(List incomingDependencyManagement, List pomAncestry) throws MavenDownloadingException {
Pom pom = pomAncestry.get(0);
if (!incomingDependencyManagement.isEmpty()) {
if (dependencyManagement == null || dependencyManagement.isEmpty()) {
dependencyManagement = new ArrayList<>();
}
for (ManagedDependency d : incomingDependencyManagement) {
if (d instanceof Imported) {
GroupArtifactVersion groupArtifactVersion = getValues(((Imported) d).getGav());
if (isAlreadyResolved(groupArtifactVersion, pomAncestry)) {
continue;
}
ResolvedPom bom = downloader.download(groupArtifactVersion, null, ResolvedPom.this, repositories)
.resolve(activeProfiles, downloader, initialRepositories, ctx);
MavenExecutionContextView.view(ctx)
.getResolutionListener()
.bomImport(bom.getGav(), pom);
dependencyManagement.addAll(ListUtils.map(bom.getDependencyManagement(), dm -> dm
.withRequestedBom(d)
.withBomGav(bom.getGav())));
} else if (d instanceof Defined) {
Defined defined = (Defined) d;
MavenExecutionContextView.view(ctx)
.getResolutionListener()
.dependencyManagement(defined.withGav(getValues(defined.getGav())), pom);
dependencyManagement.add(new ResolvedManagedDependency(
getValues(defined.getGav()),
defined.getScope() == null ? null : Scope.fromName(getValue(defined.getScope())),
getValue(defined.getType()),
getValue(defined.getClassifier()),
ListUtils.map(defined.getExclusions(), (UnaryOperator) ResolvedPom.this::getValues),
defined,
null,
null
));
}
}
}
}
private boolean isAlreadyResolved(GroupArtifactVersion groupArtifactVersion, List pomAncestry) {
for (int i = 1; i < pomAncestry.size(); i++) { // skip current pom
Pom pom = pomAncestry.get(i);
ResolvedGroupArtifactVersion alreadyResolvedGav = pom.getGav();
if (alreadyResolvedGav.getGroupId().equals(groupArtifactVersion.getGroupId()) &&
alreadyResolvedGav.getArtifactId().equals(groupArtifactVersion.getArtifactId()) &&
alreadyResolvedGav.getVersion().equals(groupArtifactVersion.getVersion())) {
return true;
}
}
return false;
}
}
public List resolveDependencies(Scope scope, MavenPomDownloader downloader, ExecutionContext ctx) throws MavenDownloadingExceptions {
return resolveDependencies(scope, new HashMap<>(), downloader, ctx);
}
public List resolveDependencies(Scope scope, Map requirements,
MavenPomDownloader downloader, ExecutionContext ctx) throws MavenDownloadingExceptions {
List dependencies = new ArrayList<>();
List dependenciesAtDepth = new ArrayList<>();
for (Dependency requestedDependency : getRequestedDependencies()) {
Dependency d = getValues(requestedDependency, 0);
Scope dScope = Scope.fromName(d.getScope());
if (dScope == scope || dScope.transitiveOf(scope) == scope) {
dependenciesAtDepth.add(new DependencyAndDependent(requestedDependency, Scope.Compile, null, requestedDependency, this));
}
}
MavenDownloadingExceptions exceptions = null;
int depth = 0;
while (!dependenciesAtDepth.isEmpty()) {
List dependenciesAtNextDepth = new ArrayList<>();
for (DependencyAndDependent dd : dependenciesAtDepth) {
// First get the dependency (relative to the pom it was defined in)
// Depth 0 prevents its dependency management from overriding versions of its own direct dependencies
Dependency d = dd.getDefinedIn().getValues(dd.getDependency(), 0);
// The dependency may be modified by the current pom's dependency management
d = getValues(d, depth);
try {
if (d.getVersion() == null) {
throw new MavenDownloadingException("No version provided", null, dd.getDependency().getGav());
}
if (d.getType() != null && (!"jar".equals(d.getType()) && !"pom".equals(d.getType()))) {
continue;
}
GroupArtifact ga = new GroupArtifact(d.getGroupId() == null ? "" : d.getGroupId(), d.getArtifactId());
VersionRequirement existingRequirement = requirements.get(ga);
if (existingRequirement == null) {
VersionRequirement newRequirement = VersionRequirement.fromVersion(d.getVersion(), depth);
requirements.put(ga, newRequirement);
String newRequiredVersion = newRequirement.resolve(ga, downloader, getRepositories());
if (newRequiredVersion == null) {
throw new MavenParsingException("Could not resolve version for [" + ga + "] matching version requirements " + newRequirement);
}
d = d.withGav(d.getGav().withVersion(newRequiredVersion));
} else {
VersionRequirement newRequirement = existingRequirement.addRequirement(d.getVersion());
requirements.put(ga, newRequirement);
String existingRequiredVersion = existingRequirement.resolve(ga, downloader, getRepositories());
String newRequiredVersion = newRequirement.resolve(ga, downloader, getRepositories());
if (newRequiredVersion == null) {
throw new MavenParsingException("Could not resolve version for [" + ga + "] matching version requirements " + newRequirement);
}
d = d.withGav(d.getGav().withVersion(newRequiredVersion));
if (!Objects.equals(existingRequiredVersion, newRequiredVersion)) {
// start over from the top with the knowledge of this new requirement and throwing
// away any in progress resolution because this requirement could cause a change
// to just about anything we've seen to this point
MavenExecutionContextView.view(ctx)
.getResolutionListener()
.clear();
return resolveDependencies(scope, requirements, downloader, ctx);
} else if (contains(dependencies, ga, d.getClassifier())) {
// we've already resolved this previously and the requirement didn't change,
// so just skip and continue on
continue;
}
}
if ((d.getGav().getGroupId() != null && d.getGav().getGroupId().startsWith("${") && d.getGav().getGroupId().endsWith("}")) ||
(d.getGav().getArtifactId().startsWith("${") && d.getGav().getArtifactId().endsWith("}")) ||
(d.getGav().getVersion() != null && d.getGav().getVersion().startsWith("${") && d.getGav().getVersion().endsWith("}"))) {
throw new MavenDownloadingException("Could not resolve property", null, d.getGav());
}
Pom dPom = downloader.download(d.getGav(), null, dd.definedIn, getRepositories());
MavenPomCache cache = MavenExecutionContextView.view(ctx).getPomCache();
ResolvedPom resolvedPom = cache.getResolvedDependencyPom(dPom.getGav());
if (resolvedPom == null) {
resolvedPom = new ResolvedPom(dPom, getActiveProfiles(), emptyMap(),
emptyList(), initialRepositories, emptyList(), emptyList(), emptyList(), emptyList(), emptyList());
resolvedPom.resolver(ctx, downloader).resolveParentsRecursively(dPom);
cache.putResolvedDependencyPom(dPom.getGav(), resolvedPom);
}
ResolvedDependency resolved = new ResolvedDependency(dPom.getRepository(),
resolvedPom.getGav(), dd.getDependency(), emptyList(),
resolvedPom.getRequested().getLicenses(),
resolvedPom.getValue(dd.getDependency().getType()),
resolvedPom.getValue(dd.getDependency().getClassifier()),
Boolean.valueOf(resolvedPom.getValue(dd.getDependency().getOptional())),
depth,
emptyList());
MavenExecutionContextView.view(ctx)
.getResolutionListener()
.dependency(scope, resolved, dd.getDefinedIn());
// build link between the including dependency and this one
ResolvedDependency includedBy = dd.getDependent();
//noinspection ConstantValue
if (includedBy != null) {
if (includedBy.getDependencies().isEmpty()) {
includedBy.unsafeSetDependencies(new ArrayList<>());
}
includedBy.getDependencies().add(resolved);
}
if (dd.getScope().transitiveOf(scope) == scope) {
dependencies.add(resolved);
} else {
continue;
}
nextDependency:
for (Dependency d2 : resolvedPom.getRequestedDependencies()) {
if (d2.getGroupId() == null) {
d2 = d2.withGav(d2.getGav().withGroupId(resolvedPom.getGroupId()));
}
//noinspection ConstantValue
if (d.getExclusions() != null) {
d2 = d2.withExclusions(ListUtils.concatAll(d2.getExclusions(), d.getExclusions()));
for (GroupArtifact exclusion : d.getExclusions()) {
if (matchesGlob(getValue(d2.getGroupId()), getValue(exclusion.getGroupId())) &&
matchesGlob(getValue(d2.getArtifactId()), getValue(exclusion.getArtifactId()))) {
if (resolved.getEffectiveExclusions().isEmpty()) {
resolved.unsafeSetEffectiveExclusions(new ArrayList<>());
}
resolved.getEffectiveExclusions().add(exclusion);
continue nextDependency;
}
}
}
String optional = resolvedPom.getValue(d2.getOptional());
if (optional != null && Boolean.parseBoolean(optional.trim())) {
continue;
}
Scope d2Scope = getDependencyScope(d2, resolvedPom);
if (d2Scope.isInClasspathOf(dd.getScope())) {
dependenciesAtNextDepth.add(new DependencyAndDependent(d2, d2Scope, resolved, dd.getRootDependent(), resolvedPom));
}
}
} catch (MavenDownloadingException e) {
exceptions = MavenDownloadingExceptions.append(exceptions, e.setRoot(dd.getRootDependent().getGav()));
}
}
dependenciesAtDepth = dependenciesAtNextDepth;
depth++;
}
if (exceptions != null) {
throw exceptions;
}
return dependencies;
}
private boolean contains(List dependencies, GroupArtifact ga, @Nullable String classifier) {
for (ResolvedDependency it : dependencies) {
if (it.getGroupId().equals(ga.getGroupId()) && it.getArtifactId().equals(ga.getArtifactId()) &&
(Objects.equals(classifier, it.getClassifier()))) {
return true;
}
}
return false;
}
private Scope getDependencyScope(Dependency d2, ResolvedPom containingPom) {
Scope scopeInContainingPom;
//noinspection ConstantConditions
if (d2.getScope() != null) {
scopeInContainingPom = Scope.fromName(getValue(d2.getScope()));
} else {
scopeInContainingPom = containingPom.getManagedScope(getValue(d2.getGroupId()), getValue(d2.getArtifactId()), getValue(d2.getType()),
getValue(d2.getClassifier()));
if (scopeInContainingPom == null) {
scopeInContainingPom = Scope.Compile;
}
}
//noinspection ConstantConditions
Scope scopeInThisProject = getManagedScope(getValue(d2.getGroupId()), getValue(d2.getArtifactId()), getValue(d2.getType()),
getValue(d2.getClassifier()));
// project POM's dependency management overrules the containingPom's dependencyManagement
// IFF the dependency is in the runtime classpath of the containingPom;
// if the dependency was not already in the classpath of the containingPom, then project POM cannot override scope / "promote" it into the classpath
return scopeInThisProject == null ? scopeInContainingPom : (scopeInContainingPom.isInClasspathOf(scopeInThisProject) ? scopeInThisProject : scopeInContainingPom);
}
private Dependency getValues(Dependency dep, int depth) {
Dependency d = dep.withGav(getValues(dep.getGav()))
.withScope(getValue(dep.getScope()));
if (d.getGroupId() == null) {
return d;
}
String scope;
if (d.getScope() == null) {
Scope parsedScope = getManagedScope(d.getGroupId(), d.getArtifactId(), d.getType(), d.getClassifier());
scope = parsedScope == null ? null : parsedScope.toString().toLowerCase();
} else {
scope = getValue(d.getScope());
}
List managedExclusions = getManagedExclusions(d.getGroupId(), d.getArtifactId(), d.getType(), d.getClassifier());
if (!managedExclusions.isEmpty()) {
d = d.withExclusions(ListUtils.concatAll(d.getExclusions(), managedExclusions));
}
if (d.getClassifier() != null) {
d = d.withClassifier(getValue(d.getClassifier()));
}
if (d.getType() != null) {
d = d.withType(getValue(d.getType()));
}
String version = d.getVersion();
if (d.getVersion() == null || depth > 0) {
// dependency management overrides transitive dependency versions
version = getManagedVersion(d.getGroupId(), d.getArtifactId(), d.getType(), d.getClassifier());
if (version == null) {
version = d.getVersion();
}
}
return d
.withGav(d.getGav().withVersion(version))
.withScope(scope);
}
@Value
private static class DependencyAndDependent {
Dependency dependency;
Scope scope;
ResolvedDependency dependent;
Dependency rootDependent;
ResolvedPom definedIn;
}
}