org.openrewrite.gradle.AddDependencyVisitor 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;
import lombok.RequiredArgsConstructor;
import org.openrewrite.*;
import org.openrewrite.gradle.internal.InsertDependencyComparator;
import org.openrewrite.gradle.marker.GradleDependencyConfiguration;
import org.openrewrite.gradle.marker.GradleProject;
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.tree.J;
import org.openrewrite.java.tree.Space;
import org.openrewrite.java.tree.Statement;
import org.openrewrite.maven.MavenDownloadingException;
import org.openrewrite.maven.MavenDownloadingExceptions;
import org.openrewrite.maven.internal.MavenPomDownloader;
import org.openrewrite.maven.table.MavenMetadataFailures;
import org.openrewrite.maven.tree.*;
import org.openrewrite.semver.*;
import org.openrewrite.tree.ParseError;
import java.util.*;
import java.util.regex.Pattern;
import java.util.stream.Collectors;
import java.util.stream.Stream;
import static java.util.Collections.*;
import static java.util.Objects.requireNonNull;
@RequiredArgsConstructor
public class AddDependencyVisitor extends GroovyIsoVisitor {
private static final MethodMatcher DEPENDENCIES_DSL_MATCHER = new MethodMatcher("RewriteGradleProject dependencies(..)");
private static final GradleParser GRADLE_PARSER = GradleParser.builder().build();
private final String groupId;
private final String artifactId;
@Nullable
private final String version;
@Nullable
private final String versionPattern;
private final String configuration;
@Nullable
private final String classifier;
@Nullable
private final String extension;
@Nullable
private final Pattern familyRegex;
@Nullable
private final MavenMetadataFailures metadataFailures;
@Nullable
private String resolvedVersion;
@Override
public G.CompilationUnit visitCompilationUnit(G.CompilationUnit cu, ExecutionContext ctx) {
Optional maybeGp = cu.getMarkers().findFirst(GradleProject.class);
if (!maybeGp.isPresent()) {
return cu;
}
GradleProject gp = maybeGp.get();
GradleDependencyConfiguration gdc = gp.getConfiguration(configuration);
if (gdc == null || gdc.findRequestedDependency(groupId, artifactId) != null) {
return cu;
}
G.CompilationUnit g = cu;
boolean dependenciesBlockMissing = true;
for (Statement statement : g.getStatements()) {
if (statement instanceof J.MethodInvocation && DEPENDENCIES_DSL_MATCHER.matches((J.MethodInvocation) statement)) {
dependenciesBlockMissing = false;
}
}
if (dependenciesBlockMissing) {
Statement dependenciesInvocation = GRADLE_PARSER.parse("dependencies {}")
.findFirst()
.map(G.CompilationUnit.class::cast)
.orElseThrow(() -> new IllegalArgumentException("Could not parse as Gradle"))
.getStatements().get(0);
dependenciesInvocation = autoFormat(dependenciesInvocation, ctx, new Cursor(getCursor(), cu));
g = g.withStatements(ListUtils.concat(g.getStatements(),
g.getStatements().isEmpty() ?
dependenciesInvocation :
dependenciesInvocation.withPrefix(Space.format("\n\n"))));
}
g = (G.CompilationUnit) new InsertDependencyInOrder(configuration, gp)
.visitNonNull(g, ctx, requireNonNull(getCursor().getParent()));
if (g != cu) {
String versionWithPattern = StringUtils.isBlank(resolvedVersion) || resolvedVersion.startsWith("$") ? null : resolvedVersion;
GradleProject newGp = addDependency(gp,
gdc,
new GroupArtifactVersion(groupId, artifactId, versionWithPattern),
classifier,
ctx);
g = g.withMarkers(g.getMarkers().setByType(newGp));
}
return g;
}
/**
* Update the dependency model, adding the specified dependency to the specified configuration and all configurations
* which extend from it.
*
* @param gp marker with the current, pre-update dependency information
* @param configuration the configuration to add the dependency to
* @param gav the group, artifact, and version of the dependency to add
* @param classifier the classifier of the dependency to add
* @param ctx context which will be used to download the pom for the dependency
* @return a copy of gp with the dependency added
*/
static GradleProject addDependency(
GradleProject gp,
@Nullable GradleDependencyConfiguration configuration,
GroupArtifactVersion gav,
@Nullable String classifier,
ExecutionContext ctx) {
try {
if (gav.getGroupId() == null || gav.getArtifactId() == null || configuration == null) {
return gp;
}
ResolvedGroupArtifactVersion resolvedGav;
List transitiveDependencies;
if (gav.getVersion() == null) {
resolvedGav = null;
transitiveDependencies = Collections.emptyList();
} else {
MavenPomDownloader mpd = new MavenPomDownloader(ctx);
Pom pom = mpd.download(gav, null, null, gp.getMavenRepositories());
ResolvedPom resolvedPom = pom.resolve(emptyList(), mpd, gp.getMavenRepositories(), ctx);
resolvedGav = resolvedPom.getGav();
transitiveDependencies = resolvedPom.resolveDependencies(Scope.Runtime, mpd, ctx);
}
Map nameToConfiguration = gp.getNameToConfiguration();
Map newNameToConfiguration = new HashMap<>(nameToConfiguration.size());
Set configurationsToAdd = Stream.concat(
Stream.of(configuration),
gp.configurationsExtendingFrom(configuration, true).stream())
.collect(Collectors.toSet());
for (GradleDependencyConfiguration gdc : nameToConfiguration.values()) {
if (!configurationsToAdd.contains(gdc)) {
newNameToConfiguration.put(gdc.getName(), gdc);
continue;
}
GradleDependencyConfiguration newGdc = gdc;
org.openrewrite.maven.tree.Dependency newRequested = new org.openrewrite.maven.tree.Dependency(
gav, classifier, "jar", gdc.getName(), emptyList(), null);
newGdc = newGdc.withRequested(ListUtils.concat(
ListUtils.map(gdc.getRequested(), requested -> {
// Remove any existing dependency with the same group and artifact id
if (Objects.equals(requested.getGroupId(), gav.getGroupId()) && Objects.equals(requested.getArtifactId(), gav.getArtifactId())) {
return null;
}
return requested;
}),
newRequested));
if (newGdc.isCanBeResolved() && resolvedGav != null) {
newGdc = newGdc.withResolved(ListUtils.concat(
ListUtils.map(gdc.getResolved(), resolved -> {
// Remove any existing dependency with the same group and artifact id
if (Objects.equals(resolved.getGroupId(), resolvedGav.getGroupId()) && Objects.equals(resolved.getArtifactId(), resolvedGav.getArtifactId())) {
return null;
}
return resolved;
}),
new ResolvedDependency(null, resolvedGav, newRequested, transitiveDependencies,
emptyList(), "jar", classifier, null, 0, null)));
}
newNameToConfiguration.put(newGdc.getName(), newGdc);
}
gp = gp.withNameToConfiguration(newNameToConfiguration);
} catch (MavenDownloadingException | MavenDownloadingExceptions | IllegalArgumentException e) {
return gp;
}
return gp;
}
@RequiredArgsConstructor
private class InsertDependencyInOrder extends GroovyIsoVisitor {
private final String configuration;
private final GradleProject gp;
@Override
public J.MethodInvocation visitMethodInvocation(J.MethodInvocation method, ExecutionContext ctx) {
J.MethodInvocation m = super.visitMethodInvocation(method, ctx);
if (!DEPENDENCIES_DSL_MATCHER.matches(m)) {
return m;
}
J.Lambda dependenciesBlock = (J.Lambda) m.getArguments().get(0);
if (!(dependenciesBlock.getBody() instanceof J.Block)) {
return m;
}
if (version != null) {
try {
resolvedVersion = resolveDependencyVersion(groupId, artifactId, "0", version, versionPattern, gp.getMavenRepositories(), metadataFailures, ctx)
.orElse(null);
} catch (MavenDownloadingException e) {
return e.warn(m);
}
}
J.Block body = (J.Block) dependenciesBlock.getBody();
String codeTemplate;
DependencyStyle style = autodetectDependencyStyle(body.getStatements());
if (style == DependencyStyle.String) {
codeTemplate = "dependencies {\n" +
escapeIfNecessary(configuration) + " \"" + groupId + ":" + artifactId + (resolvedVersion == null ? "" : ":" + resolvedVersion) + (resolvedVersion == null || classifier == null ? "" : ":" + classifier) + (extension == null ? "" : "@" + extension) + "\"" +
"\n}";
} else {
codeTemplate = "dependencies {\n" +
escapeIfNecessary(configuration) + " group: \"" + groupId + "\", name: \"" + artifactId + "\"" + (resolvedVersion == null ? "" : ", version: \"" + resolvedVersion + "\"") + (classifier == null ? "" : ", classifier: \"" + classifier + "\"") + (extension == null ? "" : ", ext: \"" + extension + "\"") +
"\n}";
}
ExecutionContext parseCtx = new InMemoryExecutionContext();
parseCtx.putMessage(ExecutionContext.REQUIRE_PRINT_EQUALS_INPUT, false);
SourceFile parsed = GRADLE_PARSER.parse(parseCtx, codeTemplate)
.findFirst()
.orElseThrow(() -> new IllegalArgumentException("Could not parse as Gradle"));
if (parsed instanceof ParseError) {
throw ((ParseError) parsed).toException();
}
J.MethodInvocation addDependencyInvocation = requireNonNull((J.MethodInvocation) ((J.Return) (((J.Block) ((J.Lambda) ((J.MethodInvocation)
((G.CompilationUnit) parsed).getStatements().get(0)).getArguments().get(0)).getBody()).getStatements().get(0))).getExpression());
addDependencyInvocation = autoFormat(addDependencyInvocation, ctx, new Cursor(getCursor(), body));
InsertDependencyComparator dependencyComparator = new InsertDependencyComparator(body.getStatements(), addDependencyInvocation);
List statements = new ArrayList<>(body.getStatements());
int i = 0;
for (; i < body.getStatements().size(); i++) {
Statement currentStatement = body.getStatements().get(i);
if (dependencyComparator.compare(currentStatement, addDependencyInvocation) > 0) {
if (dependencyComparator.getBeforeDependency() != null) {
J.MethodInvocation beforeDependency = (J.MethodInvocation) (dependencyComparator.getBeforeDependency() instanceof J.Return ?
requireNonNull(((J.Return) dependencyComparator.getBeforeDependency()).getExpression()) :
dependencyComparator.getBeforeDependency());
if (i == 0) {
if (!addDependencyInvocation.getSimpleName().equals(beforeDependency.getSimpleName())) {
statements.set(i, currentStatement.withPrefix(Space.format("\n\n" + currentStatement.getPrefix().getIndent())));
}
} else {
Space originalPrefix = addDependencyInvocation.getPrefix();
if (currentStatement instanceof J.VariableDeclarations) {
J.VariableDeclarations variableDeclarations = (J.VariableDeclarations) currentStatement;
if (variableDeclarations.getTypeExpression() != null) {
addDependencyInvocation = addDependencyInvocation.withPrefix(variableDeclarations.getTypeExpression().getPrefix());
}
} else {
addDependencyInvocation = addDependencyInvocation.withPrefix(currentStatement.getPrefix());
}
if (addDependencyInvocation.getSimpleName().equals(beforeDependency.getSimpleName())) {
if (currentStatement instanceof J.VariableDeclarations) {
J.VariableDeclarations variableDeclarations = (J.VariableDeclarations) currentStatement;
if (variableDeclarations.getTypeExpression() != null && !variableDeclarations.getTypeExpression().getPrefix().equals(originalPrefix)) {
statements.set(i, variableDeclarations.withTypeExpression(variableDeclarations.getTypeExpression().withPrefix(originalPrefix)));
}
} else if (!currentStatement.getPrefix().equals(originalPrefix)) {
statements.set(i, currentStatement.withPrefix(originalPrefix));
}
}
}
}
statements.add(i, addDependencyInvocation);
break;
}
}
if (body.getStatements().size() == i) {
if (!body.getStatements().isEmpty()) {
Statement lastStatement;
if (statements.get(i - 1) instanceof J.Return) {
J.Return r = (J.Return) statements.remove(i - 1);
lastStatement = requireNonNull(r.getExpression()).withPrefix(r.getPrefix());
statements.add(lastStatement);
} else {
lastStatement = statements.get(i - 1);
}
if (lastStatement instanceof J.MethodInvocation && !((J.MethodInvocation) lastStatement).getSimpleName().equals(addDependencyInvocation.getSimpleName())) {
addDependencyInvocation = addDependencyInvocation.withPrefix(Space.format("\n\n" + addDependencyInvocation.getPrefix().getIndent()));
}
}
statements.add(addDependencyInvocation);
}
body = body.withStatements(statements);
m = m.withArguments(singletonList(dependenciesBlock.withBody(body)));
return m;
}
}
private String escapeIfNecessary(String configurationName) {
// default is a gradle configuration created by the base plugin and a groovy keyword if
// it is used it needs to be escaped
return configurationName.equals("default") ? "'" + configurationName + "'" : configurationName;
}
enum DependencyStyle {
Map, String
}
private DependencyStyle autodetectDependencyStyle(List statements) {
int string = 0;
int map = 0;
for (Statement statement : statements) {
if (statement instanceof J.Return && ((J.Return) statement).getExpression() instanceof J.MethodInvocation) {
J.MethodInvocation invocation = (J.MethodInvocation) ((J.Return) statement).getExpression();
if (invocation.getArguments().get(0) instanceof J.Literal || invocation.getArguments().get(0) instanceof G.GString) {
string++;
} else if (invocation.getArguments().get(0) instanceof G.MapEntry) {
map++;
}
} else if (statement instanceof J.MethodInvocation) {
J.MethodInvocation invocation = (J.MethodInvocation) statement;
if (invocation.getArguments().get(0) instanceof J.Literal || invocation.getArguments().get(0) instanceof G.GString) {
string++;
} else if (invocation.getArguments().get(0) instanceof G.MapEntry) {
map++;
}
}
}
return string >= map ? DependencyStyle.String : DependencyStyle.Map;
}
public static Optional resolveDependencyVersion(String groupId, String artifactId, String currentVersion, @Nullable String newVersion, @Nullable String versionPattern,
List repositories, @Nullable MavenMetadataFailures metadataFailures, ExecutionContext ctx) throws MavenDownloadingException {
VersionComparator versionComparator = StringUtils.isBlank(newVersion) ?
new LatestRelease(versionPattern) :
requireNonNull(Semver.validate(newVersion, versionPattern).getValue());
Optional version;
if (versionComparator instanceof ExactVersion) {
version = Optional.of(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(groupId, artifactId, currentVersion, versionComparator, repositories, metadataFailures, ctx);
}
return version;
}
private static Optional findNewerVersion(String groupId, String artifactId, String version, VersionComparator versionComparator,
List repositories, @Nullable MavenMetadataFailures metadataFailures, ExecutionContext ctx) throws MavenDownloadingException {
try {
MavenMetadata mavenMetadata = metadataFailures == null ?
downloadMetadata(groupId, artifactId, repositories, ctx) :
metadataFailures.insertRows(ctx, () -> 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);
}
}