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

org.openrewrite.maven.search.DependencyInsight Maven / Gradle / Ivy

There is a newer version: 8.42.0
Show newest version
/*
 * Copyright 2020 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.maven.search; import lombok.EqualsAndHashCode; import lombok.Value; import org.jspecify.annotations.Nullable; import org.openrewrite.*; import org.openrewrite.internal.StringUtils; import org.openrewrite.java.marker.JavaProject; import org.openrewrite.java.marker.JavaSourceSet; import org.openrewrite.marker.Markup; import org.openrewrite.marker.SearchResult; import org.openrewrite.maven.MavenIsoVisitor; import org.openrewrite.maven.table.DependenciesInUse; import org.openrewrite.maven.tree.ResolvedDependency; import org.openrewrite.maven.tree.Scope; import org.openrewrite.semver.Semver; import org.openrewrite.semver.VersionComparator; import org.openrewrite.xml.tree.Xml; import java.util.Optional; /** * Find direct and transitive dependencies, marking first order dependencies that * either match or transitively include a dependency matching {@link #groupIdPattern} and * {@link #artifactIdPattern}. */ @EqualsAndHashCode(callSuper = false) @Value public class DependencyInsight extends Recipe { transient DependenciesInUse dependenciesInUse = new DependenciesInUse(this); @Option(displayName = "Group pattern", description = "Group glob pattern used to match dependencies.", example = "com.fasterxml.jackson.module") String groupIdPattern; @Option(displayName = "Artifact pattern", description = "Artifact glob pattern used to match dependencies.", example = "jackson-module-*") String artifactIdPattern; @Option(displayName = "Scope", description = "Match dependencies with the specified scope. All scopes are searched by default.", valid = {"compile", "test", "runtime", "provided", "system"}, example = "compile", required = false) @Nullable String scope; @Option(displayName = "Version", description = "Match only dependencies with the specified version. " + "Node-style [version selectors](https://docs.openrewrite.org/reference/dependency-version-selectors) may be used." + "All versions are searched by default.", example = "1.x", required = false) @Nullable String version; @Option(displayName = "Only direct", description = "If enabled, transitive dependencies will not be considered. All dependencies are searched by default.", required = false, example = "true") @Nullable Boolean onlyDirect; @Override public Validated validate() { Validated v = super.validate() .and(Validated.test("scope", "scope is a valid Maven scope", scope, s -> Scope.fromName(s) != Scope.Invalid)); if (version != null) { v = v.and(Semver.validate(version, null)); } return v; } @Override public String getDisplayName() { return "Maven dependency insight"; } @Override public String getInstanceNameSuffix() { return String.format("`%s:%s`", groupIdPattern, artifactIdPattern); } @Override public String getDescription() { return "Find direct and transitive dependencies matching a group, artifact, and scope. " + "Results include dependencies that either directly match or transitively include a matching dependency."; } @Override public TreeVisitor getVisitor() { Scope aScope = (scope == null) ? null : Scope.fromName(scope); return new MavenIsoVisitor() { @Override public Xml.Tag visitTag(Xml.Tag tag, ExecutionContext ctx) { Xml.Tag t = super.visitTag(tag, ctx); if(!isDependencyTag()) { return t; } ResolvedDependency dependency = findDependency(t, aScope); if(dependency == null) { return t; } ResolvedDependency match = dependency.findDependency(groupIdPattern, artifactIdPattern); if(match == null) { return t; } if(version != null) { VersionComparator versionComparator = Semver.validate(version, null).getValue(); if(versionComparator == null) { t = Markup.warn(t, new IllegalArgumentException("Could not construct a valid version comparator from " + version + ".")); } else { if(!versionComparator.isValid(null, match.getVersion())) { return t; } } } if (match == dependency) { t = SearchResult.found(t); } else if (Boolean.TRUE.equals(onlyDirect)) { return t; } else { t = SearchResult.found(t, match.getGav().toString()); } Optional javaProject = getCursor().firstEnclosingOrThrow(Xml.Document.class).getMarkers() .findFirst(JavaProject.class); Optional javaSourceSet = getCursor().firstEnclosingOrThrow(Xml.Document.class).getMarkers() .findFirst(JavaSourceSet.class); dependenciesInUse.insertRow(ctx, new DependenciesInUse.Row( javaProject.map(JavaProject::getProjectName).orElse(""), javaSourceSet.map(JavaSourceSet::getName).orElse("main"), match.getGroupId(), match.getArtifactId(), match.getVersion(), match.getDatedSnapshotVersion(), StringUtils.isBlank(match.getRequested().getScope()) ? "compile" : match.getRequested().getScope(), match.getDepth() )); return t; } }; } }