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

org.openrewrite.gradle.plugins.UpgradePluginVersion Maven / Gradle / Ivy

There is a newer version: 8.42.0
Show newest version
/*
 * 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.gradle.plugins; import lombok.EqualsAndHashCode; import lombok.Value; import org.openrewrite.*; import org.openrewrite.gradle.DependencyVersionSelector; import org.openrewrite.gradle.IsBuildGradle; import org.openrewrite.gradle.IsSettingsGradle; import org.openrewrite.gradle.marker.GradleProject; import org.openrewrite.gradle.marker.GradleSettings; import org.openrewrite.gradle.util.ChangeStringLiteral; import org.openrewrite.groovy.GroovyVisitor; import org.openrewrite.groovy.tree.G; import org.openrewrite.internal.ListUtils; import org.openrewrite.internal.StringUtils; import org.openrewrite.internal.lang.Nullable; import org.openrewrite.java.MethodMatcher; import org.openrewrite.java.tree.Expression; import org.openrewrite.java.tree.J; import org.openrewrite.maven.MavenDownloadingException; import org.openrewrite.maven.tree.GroupArtifact; import org.openrewrite.maven.tree.GroupArtifactVersion; import org.openrewrite.properties.PropertiesVisitor; import org.openrewrite.properties.tree.Properties; import org.openrewrite.semver.Semver; import org.openrewrite.semver.VersionComparator; import java.util.HashMap; import java.util.List; import java.util.Map; import java.util.Optional; import static java.util.Collections.singletonList; @Value @EqualsAndHashCode(callSuper = false) public class UpgradePluginVersion extends ScanningRecipe { private static final String GRADLE_PROPERTIES_FILE_NAME = "gradle.properties"; @Option(displayName = "Plugin id", description = "The `ID` part of `plugin { ID }`, as a glob expression.", example = "com.jfrog.bintray") String pluginIdPattern; @Option(displayName = "New version", description = "An exact version number or node-style semver selector used to select the version number. " + "You can also use `latest.release` for the latest available version and `latest.patch` if " + "the current version is a valid semantic version. For more details, you can look at the documentation " + "page of [version selectors](https://docs.openrewrite.org/reference/dependency-version-selectors). " + "Defaults to `latest.release`.", 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; @Override public String getDisplayName() { return "Update a Gradle plugin by id"; } @Override public String getDescription() { return "Update a Gradle plugin by id to a later version."; } @Override public Validated validate() { Validated validated = super.validate(); if (newVersion != null) { validated = validated.and(Semver.validate(newVersion, versionPattern)); } return validated; } public static class DependencyVersionState { Map versionPropNameToPluginId = new HashMap<>(); Map pluginIdToNewVersion = new HashMap<>(); } @Override public DependencyVersionState getInitialValue(ExecutionContext ctx) { return new DependencyVersionState(); } @Override public TreeVisitor getScanner(DependencyVersionState acc) { MethodMatcher pluginMatcher = new MethodMatcher("PluginSpec id(..)", false); MethodMatcher versionMatcher = new MethodMatcher("Plugin version(..)", false); GroovyVisitor groovyVisitor = new GroovyVisitor() { @Nullable private GradleProject gradleProject; @Nullable private GradleSettings gradleSettings; @Override public J visitCompilationUnit(G.CompilationUnit cu, ExecutionContext ctx) { gradleProject = cu.getMarkers().findFirst(GradleProject.class).orElse(null); gradleSettings = cu.getMarkers().findFirst(GradleSettings.class).orElse(null); if (gradleProject == null && gradleSettings == null) { return cu; } return super.visitCompilationUnit(cu, ctx); } @Override public J visitMethodInvocation(J.MethodInvocation method, ExecutionContext ctx) { J.MethodInvocation m = (J.MethodInvocation) super.visitMethodInvocation(method, ctx); if (!(versionMatcher.matches(m) && m.getSelect() instanceof J.MethodInvocation && pluginMatcher.matches(m.getSelect()))) { return m; } List pluginArgs = ((J.MethodInvocation) m.getSelect()).getArguments(); if (!(pluginArgs.get(0) instanceof J.Literal)) { return m; } String pluginId = (String) ((J.Literal) pluginArgs.get(0)).getValue(); if (pluginId == null || !StringUtils.matchesGlob(pluginId, pluginIdPattern)) { return m; } List versionArgs = m.getArguments(); try { if (versionArgs.get(0) instanceof J.Literal) { String currentVersion = (String) ((J.Literal) versionArgs.get(0)).getValue(); if (currentVersion == null) { return m; } String resolvedVersion = new DependencyVersionSelector(null, gradleProject, gradleSettings) .select(new GroupArtifactVersion(pluginId, pluginId + ".gradle.plugin", currentVersion), "classpath", newVersion, versionPattern, ctx); acc.pluginIdToNewVersion.put(pluginId, resolvedVersion); } else if (versionArgs.get(0) instanceof G.GString) { G.GString gString = (G.GString) versionArgs.get(0); if (gString == null || gString.getStrings().isEmpty() || !(gString.getStrings().get(0) instanceof G.GString.Value)) { return m; } G.GString.Value gStringValue = (G.GString.Value) gString.getStrings().get(0); String versionVariableName = gStringValue.getTree().toString(); String resolvedPluginVersion = new DependencyVersionSelector(null, gradleProject, gradleSettings) .select(new GroupArtifact(pluginId, pluginId + ".gradle.plugin"), "classpath", newVersion, versionPattern, ctx); acc.versionPropNameToPluginId.put(versionVariableName, pluginId); acc.pluginIdToNewVersion.put(pluginId, resolvedPluginVersion); } } catch (MavenDownloadingException e) { // continue } return m; } }; return Preconditions.check(Preconditions.or(new IsBuildGradle<>(), new IsSettingsGradle<>()), groovyVisitor); } @Override public TreeVisitor getVisitor(DependencyVersionState acc) { MethodMatcher pluginMatcher = new MethodMatcher("PluginSpec id(..)", false); MethodMatcher versionMatcher = new MethodMatcher("Plugin version(..)", false); PropertiesVisitor propertiesVisitor = new PropertiesVisitor() { @Override public boolean isAcceptable(SourceFile sourceFile, ExecutionContext ctx) { return super.isAcceptable(sourceFile, ctx) && sourceFile.getSourcePath().endsWith(GRADLE_PROPERTIES_FILE_NAME); } @Override public Properties visitEntry(Properties.Entry entry, ExecutionContext ctx) { if (acc.versionPropNameToPluginId.containsKey(entry.getKey())) { String currentVersion = entry.getValue().getText(); String pluginId = acc.versionPropNameToPluginId.get(entry.getKey()); if (!StringUtils.isBlank(newVersion)) { String resolvedVersion = acc.pluginIdToNewVersion.get(pluginId); VersionComparator versionComparator = Semver.validate(newVersion, versionPattern).getValue(); if (versionComparator == null) { return entry; } Optional finalVersion = versionComparator.upgrade(currentVersion, singletonList(resolvedVersion)); if (finalVersion.isPresent()) { return entry.withValue(entry.getValue().withText(resolvedVersion)); } } } return entry; } }; GroovyVisitor groovyVisitor = new GroovyVisitor() { @Override public J visitMethodInvocation(J.MethodInvocation method, ExecutionContext ctx) { J.MethodInvocation m = (J.MethodInvocation) super.visitMethodInvocation(method, ctx); if (!(versionMatcher.matches(m) && m.getSelect() instanceof J.MethodInvocation && pluginMatcher.matches(m.getSelect()))) { return m; } List pluginArgs = ((J.MethodInvocation) m.getSelect()).getArguments(); if (!(pluginArgs.get(0) instanceof J.Literal)) { return m; } String pluginId = (String) ((J.Literal) pluginArgs.get(0)).getValue(); if (pluginId == null || !StringUtils.matchesGlob(pluginId, pluginIdPattern)) { return m; } List versionArgs = m.getArguments(); if (!(versionArgs.get(0) instanceof J.Literal)) { return m; } String currentVersion = (String) ((J.Literal) versionArgs.get(0)).getValue(); if (currentVersion == null) { return m; } String resolvedVersion = acc.pluginIdToNewVersion.get(pluginId); if (resolvedVersion == null) { return m; } return m.withArguments(ListUtils.map(versionArgs, v -> ChangeStringLiteral.withStringValue((J.Literal) v, resolvedVersion))); } }; return Preconditions.or(propertiesVisitor, Preconditions.check(Preconditions.or(new IsBuildGradle<>(), new IsSettingsGradle<>()), groovyVisitor)); } }