All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.openrewrite.maven.ChangeDependencyGroupIdAndArtifactId 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; import com.fasterxml.jackson.annotation.JsonCreator; import lombok.EqualsAndHashCode; import lombok.Value; import org.jspecify.annotations.Nullable; import org.openrewrite.*; import org.openrewrite.maven.table.MavenMetadataFailures; import org.openrewrite.maven.tree.*; import org.openrewrite.semver.Semver; import org.openrewrite.semver.VersionComparator; import org.openrewrite.xml.AddToTagVisitor; import org.openrewrite.xml.ChangeTagValueVisitor; import org.openrewrite.xml.RemoveContentVisitor; import org.openrewrite.xml.tree.Xml; import java.util.*; import static org.openrewrite.Validated.required; import static org.openrewrite.Validated.test; import static org.openrewrite.internal.StringUtils.isBlank; @Value @EqualsAndHashCode(callSuper = false) public class ChangeDependencyGroupIdAndArtifactId extends Recipe { transient MavenMetadataFailures metadataFailures = new MavenMetadataFailures(this); @Option(displayName = "Old groupId", description = "The old groupId to replace. The groupId is the first part of a dependency coordinate `com.google.guava:guava:VERSION`. Supports glob expressions.", example = "org.openrewrite.recipe") String oldGroupId; @Option(displayName = "Old artifactId", description = "The old artifactId to replace. The artifactId is the second part of a dependency coordinate `com.google.guava:guava:VERSION`. Supports glob expressions.", example = "rewrite-testing-frameworks") String oldArtifactId; @Option(displayName = "New groupId", description = "The new groupId to use. Defaults to the existing group id.", example = "corp.internal.openrewrite.recipe", required = false) @Nullable String newGroupId; @Option(displayName = "New artifactId", description = "The new artifactId to use. Defaults to the existing artifact id.", example = "rewrite-testing-frameworks", required = false) @Nullable String newArtifactId; @Option(displayName = "New version", description = "An exact version number or node-style semver selector used to select the version number.", example = "29.X", required = false) @Nullable String newVersion; @Option(displayName = "Version pattern", description = "Allows version selection to be extended beyond the original Node Semver semantics. So for example," + "Setting 'version' to \"25-29\" can be paired with a metadata pattern of \"-jre\" to select Guava 29.0-jre", example = "-jre", required = false) @Nullable String versionPattern; @Option(displayName = "Override managed version", description = "If the new dependency has a managed version, this flag can be used to explicitly set the version on the dependency. The default for this flag is `false`.", required = false) @Nullable Boolean overrideManagedVersion; @Option(displayName = "Update dependency management", description = "Also update the dependency management section. The default for this flag is `true`.", required = false) @Nullable Boolean changeManagedDependency; public ChangeDependencyGroupIdAndArtifactId(String oldGroupId, String oldArtifactId, @Nullable String newGroupId, @Nullable String newArtifactId, @Nullable String newVersion, @Nullable String versionPattern) { this(oldGroupId, oldArtifactId, newGroupId, newArtifactId, newVersion, versionPattern, false, true); } @JsonCreator public ChangeDependencyGroupIdAndArtifactId(String oldGroupId, String oldArtifactId, @Nullable String newGroupId, @Nullable String newArtifactId, @Nullable String newVersion, @Nullable String versionPattern, @Nullable Boolean overrideManagedVersion, @Nullable Boolean changeManagedDependency) { this.oldGroupId = oldGroupId; this.oldArtifactId = oldArtifactId; this.newGroupId = newGroupId; this.newArtifactId = newArtifactId; this.newVersion = newVersion; this.versionPattern = versionPattern; this.overrideManagedVersion = overrideManagedVersion; this.changeManagedDependency = changeManagedDependency; } @Override public String getDisplayName() { return "Change Maven dependency"; } @Override public String getInstanceNameSuffix() { return String.format("`%s:%s`", oldGroupId, oldArtifactId); } @Override public String getDescription() { return "Change a Maven dependency coordinates. The `newGroupId` or `newArtifactId` **MUST** be different from before. " + "Matching `` coordinates are also updated if a `newVersion` or `versionPattern` is provided."; } @Override public Validated validate() { Validated validated = super.validate(); if (newVersion != null) { validated = validated.and(Semver.validate(newVersion, versionPattern)); } validated = validated.and(required("newGroupId", newGroupId).or(required("newArtifactId", newArtifactId))); validated = validated.and(test( "coordinates", "newGroupId OR newArtifactId must be different from before", this, r -> { boolean sameGroupId = isBlank(r.newGroupId) || Objects.equals(r.oldGroupId, r.newGroupId); boolean sameArtifactId = isBlank(r.newArtifactId) || Objects.equals(r.oldArtifactId, r.newArtifactId); return !(sameGroupId && sameArtifactId); } )); return validated; } @Override public TreeVisitor getVisitor() { return new MavenVisitor() { @Nullable final VersionComparator versionComparator = newVersion != null ? Semver.validate(newVersion, versionPattern).getValue() : null; @Nullable private Collection availableVersions; private boolean isNewDependencyPresent; @Override public Xml visitDocument(Xml.Document document, ExecutionContext ctx) { isNewDependencyPresent = checkIfNewDependencyPresents(newGroupId, newArtifactId, newVersion); // Any managed dependency change is unlikely to use the same version, so only update selectively. if ((changeManagedDependency == null || changeManagedDependency) && newVersion != null || versionPattern != null) { doAfterVisit(new ChangeManagedDependencyGroupIdAndArtifactId( oldGroupId, oldArtifactId, Optional.ofNullable(newGroupId).orElse(oldGroupId), Optional.ofNullable(newArtifactId).orElse(oldArtifactId), newVersion, versionPattern).getVisitor()); } return super.visitDocument(document, ctx); } @Override public Xml visitTag(Xml.Tag tag, ExecutionContext ctx) { Xml.Tag t = (Xml.Tag) super.visitTag(tag, ctx); boolean isOldDependencyTag = isDependencyTag(oldGroupId, oldArtifactId); if (isOldDependencyTag && isNewDependencyPresent) { doAfterVisit(new RemoveContentVisitor<>(tag, true, true)); maybeUpdateModel(); return t; } if (isOldDependencyTag) { String groupId = newGroupId; if (groupId != null) { t = changeChildTagValue(t, "groupId", groupId, ctx); } else { groupId = t.getChildValue("groupId").orElseThrow(NoSuchElementException::new); } String artifactId = newArtifactId; if (artifactId != null) { t = changeChildTagValue(t, "artifactId", artifactId, ctx); } else { artifactId = t.getChildValue("artifactId").orElseThrow(NoSuchElementException::new); } String currentVersion = t.getChildValue("version").orElse(null); if (newVersion != null) { try { String resolvedNewVersion = resolveSemverVersion(ctx, groupId, artifactId, currentVersion); Optional scopeTag = t.getChild("scope"); Scope scope = scopeTag.map(xml -> Scope.fromName(xml.getValue().orElse("compile"))).orElse(Scope.Compile); Optional versionTag = t.getChild("version"); boolean configuredToOverrideManageVersion = overrideManagedVersion != null && overrideManagedVersion; // False by default boolean configuredToChangeManagedDependency = changeManagedDependency == null || changeManagedDependency; // True by default boolean versionTagPresent = versionTag.isPresent(); boolean oldDependencyManaged = isDependencyManaged(scope, oldGroupId, oldArtifactId); boolean newDependencyManaged = isDependencyManaged(scope, groupId, artifactId); if (versionTagPresent) { // If the previous dependency had a version but the new artifact is managed, removed the version tag. if (!configuredToOverrideManageVersion && newDependencyManaged || (oldDependencyManaged && configuredToChangeManagedDependency)) { t = (Xml.Tag) new RemoveContentVisitor<>(versionTag.get(), false, true).visit(t, ctx); } else { // Otherwise, change the version to the new value. t = changeChildTagValue(t, "version", resolvedNewVersion, ctx); } } else if (configuredToOverrideManageVersion || !newDependencyManaged) { //If the version is not present, add the version if we are explicitly overriding a managed version or if no managed version exists. Xml.Tag newVersionTag = Xml.Tag.build("" + resolvedNewVersion + ""); //noinspection ConstantConditions t = (Xml.Tag) new AddToTagVisitor(t, newVersionTag, new MavenTagInsertionComparator(t.getChildren())).visitNonNull(t, ctx, getCursor().getParent()); } } catch (MavenDownloadingException e) { return e.warn(tag); } } if (t != tag) { maybeUpdateModel(); } } //noinspection ConstantConditions return t; } private boolean checkIfNewDependencyPresents(@Nullable String groupId, @Nullable String artifactId, @Nullable String version) { if ((groupId == null) || (artifactId == null)) { return false; } List dependencies = findDependencies(groupId, artifactId); return dependencies.stream() .filter(ResolvedDependency::isDirect) .anyMatch(rd -> (version == null) || version.equals(rd.getVersion())); } private Xml.Tag changeChildTagValue(Xml.Tag tag, String childTagName, String newValue, ExecutionContext ctx) { Optional childTag = tag.getChild(childTagName); if (childTag.isPresent() && !newValue.equals(childTag.get().getValue().orElse(null))) { tag = (Xml.Tag) new ChangeTagValueVisitor<>(childTag.get(), newValue).visitNonNull(tag, ctx); } return tag; } private boolean isDependencyManaged(Scope scope, String groupId, String artifactId) { MavenResolutionResult result = getResolutionResult(); for (ResolvedManagedDependency managedDependency : result.getPom().getDependencyManagement()) { if (groupId.equals(managedDependency.getGroupId()) && artifactId.equals(managedDependency.getArtifactId())) { return scope.isInClasspathOf(managedDependency.getScope()); } } return false; } @SuppressWarnings("ConstantConditions") private String resolveSemverVersion(ExecutionContext ctx, String groupId, String artifactId, @Nullable String currentVersion) throws MavenDownloadingException { if (versionComparator == null) { return newVersion; } String finalCurrentVersion = currentVersion != null ? currentVersion : newVersion; if (availableVersions == null) { availableVersions = new ArrayList<>(); MavenMetadata mavenMetadata = metadataFailures.insertRows(ctx, () -> downloadMetadata(groupId, artifactId, ctx)); for (String v : mavenMetadata.getVersioning().getVersions()) { if (versionComparator.isValid(finalCurrentVersion, v)) { availableVersions.add(v); } } } return availableVersions.isEmpty() ? newVersion : Collections.max(availableVersions, versionComparator); } }; } }