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

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); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy