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

org.openrewrite.yaml.format.IndentsVisitor Maven / Gradle / Ivy

There is a newer version: 8.40.3
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.yaml.format; import org.jspecify.annotations.Nullable; import org.openrewrite.Cursor; import org.openrewrite.Tree; import org.openrewrite.internal.StringUtils; import org.openrewrite.yaml.MultilineScalarChanged; import org.openrewrite.yaml.YamlIsoVisitor; import org.openrewrite.yaml.style.IndentsStyle; import org.openrewrite.yaml.tree.Yaml; import java.util.Arrays; import java.util.Iterator; import java.util.concurrent.atomic.AtomicReference; import java.util.regex.Matcher; import java.util.regex.Pattern; public class IndentsVisitor

extends YamlIsoVisitor

{ private static final Pattern LINE_BREAK = Pattern.compile("\\R"); private static final Pattern COMMENT_PATTERN = Pattern.compile("^(\\s*)(.*\\R?)", Pattern.MULTILINE); private final IndentsStyle style; @Nullable private final Tree stopAfter; public IndentsVisitor(IndentsStyle style, @Nullable Tree stopAfter) { this.style = style; this.stopAfter = stopAfter; } @Override public @Nullable Yaml visit(@Nullable Tree tree, P p, Cursor parent) { setCursor(parent); for (Cursor c = parent; c != null && c.getValue() instanceof Yaml; c = c.getParent()) { Yaml y = c.getValue(); String prefix = y.getPrefix(); if (StringUtils.hasLineBreak(prefix)) { int indent = findIndent(prefix); if (indent != 0) { c.putMessage("lastIndent", indent); } } } Iterator path = parent.getPath(Yaml.class::isInstance); if (path.hasNext()) { preVisit((Yaml) path.next(), p); } return visit(tree, p); } @Override public @Nullable Yaml preVisit(Yaml tree, P p) { if (getCursor().getNearestMessage("stop", false)) { return tree; } Yaml y = tree; int indent = getCursor().getNearestMessage("lastIndent", 0); if (StringUtils.hasLineBreak(y.getPrefix()) && !isUnindentedTopLevel()) { if (y instanceof Yaml.Sequence.Entry) { indent = getCursor().getParentOrThrow().getMessage("sequenceEntryIndent", indent); y = y.withPrefix(indentTo(y.getPrefix(), indent + style.getIndentSize())); getCursor().getParentOrThrow().putMessage("sequenceEntryIndent", indent); // the +1 is for the '-' character getCursor().getParentOrThrow().putMessage("lastIndent", indent + firstIndent(((Yaml.Sequence.Entry) y).getBlock()).length() + 1); } else if (y instanceof Yaml.Mapping.Entry) { y = y.withPrefix(indentTo(y.getPrefix(), indent + style.getIndentSize())); getCursor().putMessage("lastIndent", indent + style.getIndentSize()); } else if (y instanceof Yaml.Document) { y = y.withPrefix(indentComments(y.getPrefix(), 0)); } } else if (y instanceof Yaml.Mapping.Entry) { if (getCursor().getParentOrThrow(2).getValue() instanceof Yaml.Sequence.Entry) { // this is a mapping entry that begins a sequence entry and anything below it should be indented further to the right now, e.g.: // // - key: // value getCursor().putMessage("lastIndent", indent + style.getIndentSize()); } else { y = y.withPrefix(indentComments(y.getPrefix(), indent)); } } else if (y instanceof Yaml.Scalar && y.getMarkers().findFirst(MultilineScalarChanged.class).isPresent()) { int indentValue = indent; if (!y.getMarkers().findFirst(MultilineScalarChanged.class).get().isAdded() && indent != 0) { indentValue += style.getIndentSize(); } indentValue += y.getMarkers().findFirst(MultilineScalarChanged.class).get().getIndent(); String[] lines = LINE_BREAK.split(((Yaml.Scalar) y).getValue()); String newValue = lines[0] + ("\n" + StringUtils.trimIndent(String.join("\n", Arrays.copyOfRange(lines, 1, lines.length)))) .replaceAll("\\R", "\n" + StringUtils.repeat(" ", indentValue)); y = ((Yaml.Scalar) y).withValue(newValue); } return y; } private boolean isUnindentedTopLevel() { return getCursor().getParentOrThrow().getValue() instanceof Yaml.Document || getCursor().getParentOrThrow(2).getValue() instanceof Yaml.Document; } @Override public @Nullable Yaml postVisit(Yaml tree, P p) { if (stopAfter != null && stopAfter.isScope(tree)) { getCursor().putMessageOnFirstEnclosing(Yaml.Documents.class, "stop", true); } return super.postVisit(tree, p); } @Override public @Nullable Yaml visit(@Nullable Tree tree, P p) { if (getCursor().getNearestMessage("stop") != null) { return (Yaml) tree; } return super.visit(tree, p); } private String indentTo(String prefix, int column) { if (StringUtils.hasLineBreak(prefix)) { int indent = findIndent(prefix); if (indent != column) { int shift = column - indent; prefix = indent(prefix, shift); } } return indentComments(prefix, column); } private String indentComments(String prefix, int column) { // If the prefix contains a newline followed by a comment ensure the comment begins at the indentation column if (prefix.contains("#")) { Matcher m = COMMENT_PATTERN.matcher(prefix); StringBuilder result = new StringBuilder(); String indent = StringUtils.repeat(" ", column); boolean firstLine = true; while (m.find()) { String whitespace = m.group(1); String comment = m.group(2); int newlineCount = StringUtils.countOccurrences(whitespace, "\n"); if (firstLine && newlineCount == 0) { if (getCursor().getValue() instanceof Yaml.Documents || getCursor().getValue() instanceof Yaml.Document) { // Comments on a top-level result.append(indent); } else { // Comments can be on the end of a line and should not necessarily be moved to their own line result.append(whitespace); } result.append(comment); } else { result.append(StringUtils.repeat("\n", newlineCount)) .append(indent) .append(comment); } firstLine = false; } if (!prefix.contentEquals(result)) { prefix = result.toString(); } } return prefix; } private String indent(String whitespace, int shift) { StringBuilder newWhitespace = new StringBuilder(whitespace); shift(newWhitespace, shift); return newWhitespace.toString(); } private void shift(StringBuilder text, int shift) { if (shift > 0) { for (int i = 0; i < shift; i++) { text.append(' '); } } else { text.delete(text.length() + shift, text.length()); } } private int findIndent(String prefix) { int size = 0; for (char c : prefix.toCharArray()) { size++; if (c == '\n' || c == '\r') { size = 0; } } return size; } private String firstIndent(Yaml yaml) { AtomicReference<@Nullable String> indent = new AtomicReference<>(); new YamlIsoVisitor>() { @Override public @Nullable Yaml visit(@Nullable Tree tree, AtomicReference indent) { Yaml y = (Yaml) tree; //noinspection ConstantValue if (indent.get() != null) { return y; } if (y != null && y != yaml && !(y instanceof Yaml.Mapping) && !(y instanceof Yaml.Sequence.Entry)) { indent.set(y.getPrefix()); return y; } return super.visit(tree, indent); } }.visit(yaml, indent); String indentStr = indent.get(); return indentStr == null ? "" : indentStr; } }