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

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





© 2015 - 2025 Weber Informatics LLC | Privacy Policy