org.openrewrite.maven.search.FindProperties Maven / Gradle / Ivy
/*
* 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.ExecutionContext;
import org.openrewrite.Option;
import org.openrewrite.Recipe;
import org.openrewrite.TreeVisitor;
import org.openrewrite.internal.ListUtils;
import org.openrewrite.marker.SearchResult;
import org.openrewrite.maven.MavenVisitor;
import org.openrewrite.maven.table.MavenProperties;
import org.openrewrite.xml.tree.Content;
import org.openrewrite.xml.tree.Xml;
import java.util.*;
import java.util.regex.Pattern;
import static org.openrewrite.Tree.randomId;
@EqualsAndHashCode(callSuper = false)
@Value
public class FindProperties extends Recipe {
transient MavenProperties mavenProperties = new MavenProperties(this);
@Option(displayName = "Property pattern",
description = "Regular expression pattern used to match property tag names.",
example = "guava.*")
String propertyPattern;
@Option(displayName = "Value pattern",
description = "Regular expression pattern used to match property values.",
example = "28.*",
required = false)
@Nullable
String valuePattern;
UUID searchId = randomId();
@Override
public String getDisplayName() {
return "Find Maven project properties";
}
@Override
public String getDescription() {
return "Finds the specified Maven project properties within a pom.xml.";
}
@Override
public TreeVisitor, ExecutionContext> getVisitor() {
Pattern propertyMatcher = Pattern.compile(propertyPattern);
Pattern propertyUsageMatcher = Pattern.compile(".*\\$\\{" + propertyMatcher.pattern() + "}.*");
Pattern valueMatcher = valuePattern == null ? null : Pattern.compile(valuePattern);
return new MavenVisitor() {
@Override
public Xml visitTag(Xml.Tag tag, ExecutionContext ctx) {
Xml.Tag t = (Xml.Tag) super.visitTag(tag, ctx);
if (isPropertyTag() && propertyMatcher.matcher(t.getName()).matches()) {
if (valueMatcher == null) {
t = SearchResult.found(t);
mavenProperties.insertRow(ctx, new MavenProperties.Row(t.getName(), t.getValue().orElse(null)));
} else {
Optional value = t.getValue();
if (value.isPresent() && valueMatcher.matcher(value.get()).matches()) {
t = SearchResult.found(t);
mavenProperties.insertRow(ctx, new MavenProperties.Row(t.getName(), value.get()));
}
}
}
Optional value = t.getValue();
if (value.isPresent() && propertyUsageMatcher.matcher(value.get()).matches()) {
//noinspection unchecked
t = t.withContent(ListUtils.mapFirst((List) t.getContent(), v ->
SearchResult.found(v, getResolutionResult().getPom().getValue(value.get()))));
}
return t;
}
};
}
/**
* @param xml The xml document of the pom.xml
* @param propertyPattern Regular expression pattern used to match property tag names
* @return Set of Maven project property tags that matches the {@code propertyPattern} within a pom.xml
*/
public static Set find(Xml.Document xml, String propertyPattern) {
Pattern propertyMatcher = Pattern.compile(propertyPattern);
Set found = new HashSet<>();
new MavenVisitor>() {
@Override
public Xml visitTag(Xml.Tag tag, Set tags) {
Xml.Tag t = (Xml.Tag) super.visitTag(tag, tags);
if (isPropertyTag() && propertyMatcher.matcher(tag.getName()).matches()) {
tags.add(t);
}
return t;
}
}.visit(xml, found);
return found;
}
}