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

org.openrewrite.python.tree.PySpace Maven / Gradle / Ivy

There is a newer version: 1.17.2
Show newest version
/*
 * Copyright 2023 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.python.tree; import org.openrewrite.internal.ListUtils; import org.openrewrite.internal.lang.Nullable; import org.openrewrite.java.tree.Comment; import org.openrewrite.java.tree.Space; import org.openrewrite.marker.Markers; import java.util.ArrayList; import java.util.Collections; import java.util.List; public final class PySpace { public static final class SpaceBuilder { private @Nullable String initialWhitespace; private @Nullable List comments; private @Nullable StringBuilder whitespaceBuilder; private @Nullable String nextComment; private String finishWhitespace() { if (whitespaceBuilder == null) { return ""; } else { String ws = whitespaceBuilder.toString(); whitespaceBuilder.setLength(0); return ws; } } private void finishComment() { String whitespace = finishWhitespace(); if (nextComment != null) { if (comments == null) { comments = new ArrayList<>(); } comments.add(new PyComment(nextComment, whitespace, false, Markers.EMPTY)); } else if (!whitespace.isEmpty()) { if (this.initialWhitespace != null) { throw new IllegalStateException("unexpected"); } this.initialWhitespace = whitespace; } } @SuppressWarnings("UnusedReturnValue") public SpaceBuilder addWhitespace(String whitespace) { if (whitespaceBuilder == null) { whitespaceBuilder = new StringBuilder(); } whitespaceBuilder.append(whitespace); return this; } @SuppressWarnings("UnusedReturnValue") public SpaceBuilder addComment(String commentWithHash) { finishComment(); nextComment = commentWithHash; return this; } public Space build() { finishComment(); Space space = Space.build( initialWhitespace == null ? "" : initialWhitespace, comments == null ? Collections.emptyList() : comments ); reset(); return space; } public SpaceBuilder reset() { this.initialWhitespace = null; this.whitespaceBuilder.setLength(0); this.comments = null; return this; } } public static Space appendWhitespace(Space space, String whitespace) { if (!space.getComments().isEmpty()) { return space.withComments( ListUtils.mapLast( space.getComments(), comment -> comment.withSuffix(comment.getSuffix() + whitespace) ) ); } else { return space.withWhitespace( space.getWhitespace() + whitespace ); } } public static Space appendComment(Space space, String commentWithHash) { final String commentText = validateComment(commentWithHash); return space.withComments(ListUtils.concat( space.getComments(), new PyComment(commentText, "", false, Markers.EMPTY) )); } private static String validateComment(String commentWithHash) { if (!commentWithHash.startsWith("#")) { throw new IllegalArgumentException("comment should start with a hash"); } if (commentWithHash.contains("\n")) { throw new IllegalArgumentException("comment cannot contain newlines"); } return commentWithHash.substring(1); } public enum IndentStartMode { LINE_START, AFTER_STATEMENT, } public enum IndentEndMode { STATEMENT_START, REST_OF_LINE, } public static Space reindent(Space original, String indentWithoutNewline, IndentStartMode startMode, IndentEndMode endMode) { if (indentWithoutNewline.contains("\n")) { throw new IllegalArgumentException("argument to `deindent` should not contain newline: " + Space.build(indentWithoutNewline, Collections.emptyList())); } if (indentWithoutNewline.isEmpty()) { return original; } switch (endMode) { case REST_OF_LINE: if (!original.getLastWhitespace().endsWith("\n")) { throw new IllegalStateException("expected statement suffix to end with a newline: " + original); } break; case STATEMENT_START: if (!original.getComments().isEmpty() || original.getLastWhitespace().contains("\n")) { if (!original.getLastWhitespace().endsWith("\n")) { throw new IllegalStateException("expected statement prefix to end with an indent placeholder: " + original); } } else { if (!original.getLastWhitespace().isEmpty()) { throw new IllegalStateException("expected statement prefix to end with an indent placeholder: " + original); } } break; } Space space = Space.build(original.getWhitespace(), Collections.emptyList()); List originalComments = original.getComments(); if (original.getComments().isEmpty()) { if (startMode == IndentStartMode.LINE_START && endMode == IndentEndMode.STATEMENT_START) { if (original.getWhitespace().isEmpty() || original.getWhitespace().endsWith("\n")) { space = appendWhitespace(space, indentWithoutNewline); } } } else { for (int i = 0; i < originalComments.size(); i++) { Comment originalComment = originalComments.get(i); PyComment comment = (PyComment) originalComment; if (comment.isAlignedToIndent() && (i != 0 || startMode == IndentStartMode.LINE_START)) { if (space.getLastWhitespace().isEmpty() || space.getLastWhitespace().endsWith("\n")) { space = appendWhitespace(space, indentWithoutNewline); } } space = space.withComments(ListUtils.concat( space.getComments(), comment )); } if (endMode == IndentEndMode.STATEMENT_START) { space = appendWhitespace(space, indentWithoutNewline); } } return space; } public static Space deindent(Space original, String indentWithoutNewline, IndentStartMode startMode, IndentEndMode endMode) { if (indentWithoutNewline.contains("\n")) { throw new IllegalArgumentException("argument to `deindent` should not contain newline: " + Space.build(indentWithoutNewline, Collections.emptyList())); } if (indentWithoutNewline.isEmpty()) { return original; } final String indentWithNewline = "\n" + indentWithoutNewline; switch (endMode) { case REST_OF_LINE: if (!original.getLastWhitespace().endsWith("\n")) { throw new IllegalStateException("expected statement suffix to end with a newline"); } break; case STATEMENT_START: if (!original.getComments().isEmpty() || original.getLastWhitespace().contains("\n")) { if (!original.getLastWhitespace().endsWith(indentWithNewline)) { throw new IllegalStateException("expected statement prefix to end with an indent"); } } else { if (!original.getLastWhitespace().equals(indentWithoutNewline)) { throw new IllegalStateException("expected statement prefix to end with an indent"); } } break; } Space space; boolean currentlyIndented; if (startMode == IndentStartMode.LINE_START && original.getWhitespace().equals(indentWithoutNewline)) { if (endMode == IndentEndMode.STATEMENT_START || !original.getComments().isEmpty()) { currentlyIndented = true; space = Space.EMPTY; } else { // weird coincidence; maybe not possible? currentlyIndented = false; space = Space.build(original.getWhitespace(), Collections.emptyList()); } } else if (original.getWhitespace().endsWith(indentWithNewline)) { currentlyIndented = true; space = Space.build( original.getWhitespace().substring( // just keep the newline 0, original.getWhitespace().length() - indentWithoutNewline.length() ), Collections.emptyList() ); } else { currentlyIndented = false; space = Space.build(original.getWhitespace(), Collections.emptyList()); } List originalComments = original.getComments(); for (int i = 0; i < originalComments.size(); i++) { Comment originalComment = originalComments.get(i); PyComment comment = (PyComment) originalComment; comment = comment.withAlignedToIndent(currentlyIndented); final boolean isLastComment = i == originalComments.size() - 1; if (!isLastComment || endMode == IndentEndMode.STATEMENT_START) { currentlyIndented = comment.getSuffix().endsWith(indentWithNewline); if (currentlyIndented) { comment = comment.withSuffix( comment.getSuffix().substring( // just keep the newline 0, comment.getSuffix().length() - indentWithoutNewline.length() ) ); } } else { currentlyIndented = false; } space = space.withComments( ListUtils.concat(space.getComments(), comment) ); } return space; } public static Space stripIndent(Space space, String expectedIndent) { if (space.getComments().isEmpty()) { final String ws = space.getWhitespace(); if (!ws.endsWith(expectedIndent)) { throw new IllegalStateException("expected statement prefix to end with block indent"); } space = space.withWhitespace( ws.substring(0, ws.length() - expectedIndent.length()) ); } else { space = space.withComments( ListUtils.mapLast( space.getComments(), lastComment -> { final String suffix = lastComment.getSuffix(); if (!suffix.endsWith(expectedIndent)) { throw new IllegalStateException("expected statement prefix to end with block indent"); } return lastComment.withSuffix( suffix.substring(0, suffix.length() - expectedIndent.length()) ); } ) ); } return space; } public enum Location { ASSERT_PREFIX, ASSERT_ELEMENT_SUFFIX, AWAIT_PREFIX, COMPREHENSION_CLAUSE_PREFIX, COMPREHENSION_CONDITION_PREFIX, COMPREHENSION_IN, COMPREHENSION_PREFIX, COMPREHENSION_SUFFIX, NAMED_ARGUMENT_PREFIX, DEL_ELEMENT_SUFFIX, DEL_PREFIX, DICT_ENTRY, DICT_ENTRY_KEY_SUFFIX, DICT_LITERAL_ELEMENT_SUFFIX, DICT_LITERAL_PREFIX, ELSE_WRAPPER_PREFIX, EXCEPTION_TYPE_PREFIX, ERROR_FROM_PREFIX, ERROR_FROM_SOURCE, KEY_VALUE_PREFIX, KEY_VALUE_SUFFIX, MATCH_CASE_GUARD, MATCH_CASE_PREFIX, MATCH_PATTERN_PREFIX, MATCH_PATTERN_ELEMENT_PREFIX, MATCH_PATTERN_ELEMENT_SUFFIX, NAMED_ARGUMENT, PASS_PREFIX, SPECIAL_ARG_PREFIX, SPECIAL_PARAM_PREFIX, TOP_LEVEL_STATEMENT, TRAILING_ELSE_WRAPPER_PREFIX, TYPE_HINT_PREFIX, TYPE_HINTED_EXPRESSION_PREFIX, VARIABLE_SCOPE_NAME_SUFFIX, VARIABLE_SCOPE_PREFIX, YIELD_FROM_PREFIX, YIELD_PREFIX, YIELD_ELEMENT_SUFFIX, } private PySpace() { } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy