org.openrewrite.gradle.plugins.AddPluginVisitor Maven / Gradle / Ivy
/*
* Copyright 2022 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.ExecutionContext;
import org.openrewrite.Parser;
import org.openrewrite.gradle.GradleParser;
import org.openrewrite.gradle.search.FindPlugins;
import org.openrewrite.groovy.GroovyIsoVisitor;
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.search.FindMethods;
import org.openrewrite.java.tree.*;
import org.openrewrite.maven.MavenDownloadingException;
import org.openrewrite.maven.internal.MavenPomDownloader;
import org.openrewrite.maven.tree.GroupArtifact;
import org.openrewrite.maven.tree.MavenMetadata;
import org.openrewrite.maven.tree.MavenRepository;
import org.openrewrite.semver.*;
import org.openrewrite.tree.ParseError;
import java.nio.file.Paths;
import java.util.Collections;
import java.util.List;
import java.util.Optional;
import java.util.concurrent.atomic.AtomicInteger;
import static java.util.Collections.emptyMap;
import static java.util.Collections.singletonList;
import static java.util.Objects.requireNonNull;
@Value
@EqualsAndHashCode(callSuper = false)
public class AddPluginVisitor extends GroovyIsoVisitor {
String pluginId;
@Nullable
String newVersion;
@Nullable
String versionPattern;
List repositories;
public static Optional resolvePluginVersion(String pluginId, String currentVersion, @Nullable String newVersion, @Nullable String versionPattern,
List repositories, ExecutionContext ctx) throws MavenDownloadingException {
VersionComparator versionComparator = StringUtils.isBlank(newVersion) ?
new LatestRelease(null) :
requireNonNull(Semver.validate(newVersion, versionPattern).getValue());
Optional version;
if (versionComparator instanceof ExactVersion) {
version = versionComparator.upgrade(currentVersion, Collections.singletonList(newVersion));
} else if (versionComparator instanceof LatestPatch && !versionComparator.isValid(currentVersion, currentVersion)) {
// in the case of "latest.patch", a new version can only be derived if the
// current version is a semantic version
return Optional.empty();
} else {
version = findNewerVersion(pluginId, pluginId + ".gradle.plugin", currentVersion, versionComparator, repositories, ctx);
}
return version;
}
private static Optional findNewerVersion(String groupId, String artifactId, String version, VersionComparator versionComparator,
List repositories, ExecutionContext ctx) throws MavenDownloadingException {
try {
MavenMetadata mavenMetadata = downloadMetadata(groupId, artifactId, repositories, ctx);
return versionComparator.upgrade(version, mavenMetadata.getVersioning().getVersions());
} catch (IllegalStateException e) {
// this can happen when we encounter exotic versions
return Optional.empty();
}
}
private static MavenMetadata downloadMetadata(String groupId, String artifactId, List repositories, ExecutionContext ctx) throws MavenDownloadingException {
return new MavenPomDownloader(emptyMap(), ctx, null, null)
.downloadMetadata(new GroupArtifact(groupId, artifactId), null,
repositories);
}
private static @Nullable Comment getLicenseHeader(G.CompilationUnit cu) {
if (!cu.getStatements().isEmpty()) {
Statement firstStatement = cu.getStatements().get(0);
if (!firstStatement.getComments().isEmpty()) {
Comment firstComment = firstStatement.getComments().get(0);
if (isLicenseHeader(firstComment)) {
return firstComment;
}
}
} else if (cu.getEof() != null && !cu.getEof().getComments().isEmpty()) {
Comment firstComment = cu.getEof().getComments().get(0);
if (isLicenseHeader(firstComment)) {
// Adding suffix so when we later use it, formats well.
return firstComment.withSuffix("\n\n");
}
}
return null;
}
private static boolean isLicenseHeader(Comment comment) {
return comment instanceof TextComment && comment.isMultiline() &&
((TextComment) comment).getText().contains("License");
}
private static G.CompilationUnit removeLicenseHeader(G.CompilationUnit cu) {
if (!cu.getStatements().isEmpty()) {
return cu.withStatements(ListUtils.mapFirst(cu.getStatements(),
s -> s.withComments(s.getComments().subList(1, s.getComments().size()))
));
} else {
List eofComments = cu.getEof().getComments();
return cu.withEof(cu.getEof().withComments(eofComments.subList(1, eofComments.size())));
}
}
@Override
public G.CompilationUnit visitCompilationUnit(G.CompilationUnit cu, ExecutionContext ctx) {
if (FindPlugins.find(cu, pluginId).isEmpty()) {
Optional version;
if (newVersion == null) {
// We have been requested to add a versionless plugin
version = Optional.empty();
} else {
try {
version = resolvePluginVersion(pluginId, "0", newVersion, versionPattern, repositories, ctx);
} catch (MavenDownloadingException e) {
return e.warn(cu);
}
}
AtomicInteger singleQuote = new AtomicInteger();
AtomicInteger doubleQuote = new AtomicInteger();
new GroovyIsoVisitor() {
final MethodMatcher pluginIdMatcher = new MethodMatcher("PluginSpec id(..)");
@Override
public J.MethodInvocation visitMethodInvocation(J.MethodInvocation method, Integer integer) {
J.MethodInvocation m = super.visitMethodInvocation(method, integer);
if (pluginIdMatcher.matches(m)) {
if (m.getArguments().get(0) instanceof J.Literal) {
J.Literal l = (J.Literal) m.getArguments().get(0);
assert l.getValueSource() != null;
if (l.getValueSource().startsWith("'")) {
singleQuote.incrementAndGet();
} else {
doubleQuote.incrementAndGet();
}
}
}
return m;
}
}.visitCompilationUnit(cu, 0);
String delimiter = singleQuote.get() < doubleQuote.get() ? "\"" : "'";
String source = "plugins {\n" +
" id " + delimiter + pluginId + delimiter + (version.map(s -> " version " + delimiter + s + delimiter).orElse("")) + "\n" +
"}";
Statement statement = GradleParser.builder().build()
.parseInputs(singletonList(Parser.Input.fromString(source)), null, ctx)
.findFirst()
.map(parsed -> {
if (parsed instanceof ParseError) {
throw ((ParseError) parsed).toException();
}
return ((G.CompilationUnit) parsed);
})
.map(parsed -> parsed.getStatements().get(0))
.orElseThrow(() -> new IllegalArgumentException("Could not parse as Gradle"));
if (FindMethods.find(cu, "RewriteGradleProject plugins(..)").isEmpty() && FindMethods.find(cu, "RewriteSettings plugins(..)").isEmpty()) {
if (cu.getSourcePath().endsWith(Paths.get("settings.gradle"))
&& !cu.getStatements().isEmpty()
&& cu.getStatements().get(0) instanceof J.MethodInvocation
&& ((J.MethodInvocation) cu.getStatements().get(0)).getSimpleName().equals("pluginManagement")) {
return cu.withStatements(ListUtils.insert(cu.getStatements(), autoFormat(statement.withPrefix(Space.format("\n\n")), ctx, getCursor()), 1));
} else {
int insertAtIdx = 0;
for (int i = 0; i < cu.getStatements().size(); i++) {
Statement existingStatement = cu.getStatements().get(i);
if (existingStatement instanceof J.MethodInvocation && ((J.MethodInvocation) existingStatement).getSimpleName().equals("buildscript")) {
insertAtIdx = i + 1;
break;
}
}
if (insertAtIdx == 0) {
Comment licenseHeader = getLicenseHeader(cu);
if (licenseHeader != null) {
cu = removeLicenseHeader(cu);
statement = statement.withComments(Collections.singletonList(licenseHeader));
}
Space leadingSpace = Space.firstPrefix(cu.getStatements());
return cu.withStatements(ListUtils.insert(
Space.formatFirstPrefix(cu.getStatements(), leadingSpace.withWhitespace("\n\n" + leadingSpace.getWhitespace())),
autoFormat(statement, ctx, getCursor()),
insertAtIdx));
} else {
return cu.withStatements(ListUtils.insert(cu.getStatements(), autoFormat(statement.withPrefix(Space.format("\n\n")), ctx, getCursor()), insertAtIdx));
}
}
} else {
MethodMatcher buildPluginsMatcher = new MethodMatcher("RewriteGradleProject plugins(groovy.lang.Closure)");
MethodMatcher settingsPluginsMatcher = new MethodMatcher("RewriteSettings plugins(groovy.lang.Closure)");
J.MethodInvocation pluginDef = (J.MethodInvocation) ((J.Return) ((J.Block) ((J.Lambda) ((J.MethodInvocation) autoFormat(statement, ctx, getCursor())).getArguments().get(0)).getBody()).getStatements().get(0)).getExpression();
return cu.withStatements(ListUtils.map(cu.getStatements(), stat -> {
if (stat instanceof J.MethodInvocation) {
J.MethodInvocation m = (J.MethodInvocation) stat;
if (buildPluginsMatcher.matches(m) || settingsPluginsMatcher.matches(m)) {
m = m.withArguments(ListUtils.map(m.getArguments(), a -> {
if (a instanceof J.Lambda) {
J.Lambda l = (J.Lambda) a;
J.Block b = (J.Block) l.getBody();
List pluginStatements = b.getStatements();
if (!pluginStatements.isEmpty() && pluginStatements.get(pluginStatements.size() - 1) instanceof J.Return) {
Statement last = pluginStatements.remove(pluginStatements.size() - 1);
Expression lastExpr = requireNonNull(((J.Return) last).getExpression());
pluginStatements.add(lastExpr.withPrefix(last.getPrefix()));
}
pluginStatements.add(pluginDef);
return l.withBody(autoFormat(b.withStatements(pluginStatements), ctx, getCursor()));
}
return a;
}));
return m;
}
}
return stat;
}));
}
}
return super.visitCompilationUnit(cu, ctx);
}
}