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

org.openrewrite.test.RewriteTest Maven / Gradle / Ivy

There is a newer version: 8.42.0
Show newest version
/*
 * 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.test; import lombok.RequiredArgsConstructor; import lombok.experimental.Delegate; import org.assertj.core.api.SoftAssertions; import org.jspecify.annotations.Nullable; import org.openrewrite.*; import org.openrewrite.config.CompositeRecipe; import org.openrewrite.config.Environment; import org.openrewrite.config.OptionDescriptor; import org.openrewrite.internal.InMemoryDiffEntry; import org.openrewrite.internal.RecipeIntrospectionUtils; import org.openrewrite.internal.StringUtils; import org.openrewrite.internal.WhitespaceValidationService; import org.openrewrite.marker.Marker; import org.openrewrite.marker.Markers; import org.openrewrite.quark.Quark; import org.openrewrite.remote.Remote; import org.openrewrite.tree.ParseError; import org.openrewrite.tree.ParsingExecutionContextView; import java.nio.charset.StandardCharsets; import java.nio.file.Path; import java.util.*; import java.util.concurrent.CopyOnWriteArrayList; import java.util.function.Consumer; import java.util.function.Function; import java.util.function.Supplier; import java.util.function.UnaryOperator; import java.util.stream.Collectors; import java.util.stream.StreamSupport; import static org.assertj.core.api.Assertions.*; import static org.openrewrite.internal.StringUtils.trimIndentPreserveCRLF; @SuppressWarnings("unused") public interface RewriteTest extends SourceSpecs { static AdHocRecipe toRecipe(Supplier> visitor) { return new AdHocRecipe(null, null, null, visitor, null, null); } static AdHocRecipe toRecipe() { return new AdHocRecipe(null, null, null, TreeVisitor::noop, null, null); } static AdHocRecipe toRecipe(Function> visitor) { AdHocRecipe r = toRecipe(); return r.withGetVisitor(() -> visitor.apply(r)); } static Recipe fromRuntimeClasspath(String recipe) { return Environment.builder() .scanRuntimeClasspath() .build() .activateRecipes(recipe); } /** * Check that all recipes loadable from the runtime classpath containing the provided package name * are configurable and run with an empty source file set. * * @param packageName The package name to scan for recipes in. */ default void assertRecipesConfigure(String packageName) { // soft assertions allow the entire stack trace to be displayed for each // recipe that fails to configure SoftAssertions softly = new SoftAssertions(); for (Recipe recipe : Environment.builder() .scanRuntimeClasspath(packageName) .build() .listRecipes()) { // scanRuntimeClasspath picks up all recipes in META-INF/rewrite regardless of whether their // names start with the package we intend to filter on here if (recipe.getName().startsWith(packageName)) { softly.assertThatCode(() -> { try { rewriteRun( spec -> spec.recipe(recipe), new SourceSpecs[0] ); } catch (Throwable t) { fail("Recipe " + recipe.getName() + " failed to configure", t); } }).doesNotThrowAnyException(); } } softly.assertAll(); } /** * Check that all recipes loadable from the runtime classpath containing the test's package * are configurable and run with an empty source file set. */ default void assertRecipesConfigure() { assertRecipesConfigure(getClass().getPackage().getName()); } /** * @return always null, a method that better documents in code that a source file does not exist either * before or after a recipe run. */ default @Nullable String doesNotExist() { return null; } default void defaults(RecipeSpec spec) { spec.recipe(Recipe.noop()); } default void rewriteRun(SourceSpecs... sourceSpecs) { rewriteRun(spec -> { }, sourceSpecs); } default void rewriteRun(Consumer spec, SourceSpecs... sourceSpecs) { rewriteRun(spec, Arrays.stream(sourceSpecs) .flatMap(specGroup -> StreamSupport.stream(specGroup.spliterator(), false)) .toArray(SourceSpec[]::new) ); } default void rewriteRun(Consumer spec, SourceSpec... sourceSpecs) { RecipeSpec testClassSpec = RecipeSpec.defaults(); defaults(testClassSpec); RecipeSpec testMethodSpec = RecipeSpec.defaults(); spec.accept(testMethodSpec); PrintOutputCapture.MarkerPrinter markerPrinter; if (testMethodSpec.getMarkerPrinter() != null) { markerPrinter = testMethodSpec.getMarkerPrinter(); } else if (testClassSpec.getMarkerPrinter() != null) { markerPrinter = testClassSpec.getMarkerPrinter(); } else { markerPrinter = PrintOutputCapture.MarkerPrinter.DEFAULT; } PrintOutputCapture out = new PrintOutputCapture<>(0, markerPrinter); Recipe recipe = testMethodSpec.recipe == null ? testClassSpec.recipe == null ? Recipe.noop() : testClassSpec.recipe : testMethodSpec.recipe; if (!(recipe instanceof AdHocRecipe) && !(recipe instanceof AdHocScanningRecipe) && !(recipe instanceof CompositeRecipe) && !(recipe.equals(Recipe.noop())) && testClassSpec.serializationValidation && testMethodSpec.serializationValidation) { RecipeSerializer recipeSerializer = new RecipeSerializer(); assertThat(recipeSerializer.read(recipeSerializer.write(recipe))) .as("Recipe must be serializable/deserializable") .isEqualTo(recipe); assertThatCode(() -> { Recipe r = RecipeIntrospectionUtils.constructRecipe(recipe.getClass()); // getRecipeList should not fail with default parameters from RecipeIntrospectionUtils. r.getRecipeList(); // We add recipes to HashSet in some places, we need to validate that hashCode and equals does not fail. //noinspection ResultOfMethodCallIgnored r.hashCode(); //noinspection EqualsWithItself,ResultOfMethodCallIgnored r.equals(r); }) .as("Recipe must be able to instantiate via RecipeIntrospectionUtils") .doesNotThrowAnyException(); validateRecipeNameAndDescription(recipe); validateRecipeOptions(recipe); } if (testMethodSpec.getRecipePrinter() != null) { testMethodSpec.getRecipePrinter().printTree(recipe); } else if (testClassSpec.getRecipePrinter() != null) { testClassSpec.getRecipePrinter().printTree(recipe); } int cycles = testMethodSpec.cycles == null ? testClassSpec.getCycles() : testMethodSpec.getCycles(); // There may not be any tests that have "after" assertions, but that change file attributes or file names. // If so, the test can declare an expected set of cycles that make changes. int expectedCyclesThatMakeChanges = testMethodSpec.expectedCyclesThatMakeChanges == null ? (testClassSpec.expectedCyclesThatMakeChanges == null ? 0 : testClassSpec.expectedCyclesThatMakeChanges) : testMethodSpec.expectedCyclesThatMakeChanges; // If there are any tests that have assertions (an "after"), then set the expected cycles. for (SourceSpec s : sourceSpecs) { if (s.after != null) { expectedCyclesThatMakeChanges = testMethodSpec.expectedCyclesThatMakeChanges == null ? testClassSpec.getExpectedCyclesThatMakeChanges(cycles) : testMethodSpec.getExpectedCyclesThatMakeChanges(cycles); break; } } ExecutionContext ctx; if (testMethodSpec.getExecutionContext() != null) { ctx = testMethodSpec.getExecutionContext(); } else if (testClassSpec.getExecutionContext() != null) { ctx = testClassSpec.getExecutionContext(); } else { ctx = defaultExecutionContext(sourceSpecs); } for (SourceSpec s : sourceSpecs) { s.customizeExecutionContext.accept(ctx); } List> validations = new ArrayList<>(); recipe.validateAll(ctx, validations); assertThat(validations.stream().filter(Validated::isInvalid)) .as("Recipe validation must have no failures") .isEmpty(); Map>> sourceSpecsByParser = new HashMap<>(); List methodSpecParsers = testMethodSpec.parsers; // Clone class-level parsers to ensure that no state leaks between tests List testClassSpecParsers = testClassSpec.parsers.stream() .map(Parser.Builder::clone) .collect(Collectors.toList()); for (SourceSpec sourceSpec : sourceSpecs) { // ----- method specific parser ------------------------- if (RewriteTestUtils.groupSourceSpecsByParser(methodSpecParsers, sourceSpecsByParser, sourceSpec)) { continue; } // ----- test default parser ------------------------- if (RewriteTestUtils.groupSourceSpecsByParser(testClassSpecParsers, sourceSpecsByParser, sourceSpec)) { continue; } // ----- default parsers for each SourceFile type ------------------------- sourceSpecsByParser.computeIfAbsent(sourceSpec.getParser().clone(), p -> new ArrayList<>()).add(sourceSpec); } Map> specBySourceFile = new LinkedHashMap<>(sourceSpecs.length); for (Map.Entry>> sourceSpecsForParser : sourceSpecsByParser.entrySet()) { Map, Parser.Input> inputs = new LinkedHashMap<>(sourceSpecsForParser.getValue().size()); Parser parser = sourceSpecsForParser.getKey().build(); for (SourceSpec sourceSpec : sourceSpecsForParser.getValue()) { if (sourceSpec.before == null) { continue; } String beforeTrimmed = sourceSpec.noTrim ? sourceSpec.before : trimIndentPreserveCRLF(sourceSpec.before); Path sourcePath; if (sourceSpec.sourcePath != null) { sourcePath = sourceSpec.dir.resolve(sourceSpec.sourcePath); } else { sourcePath = parser.sourcePathFromSourceText(sourceSpec.dir, beforeTrimmed); } for (UncheckedConsumer> consumer : testMethodSpec.allSources) { consumer.accept(sourceSpec); } for (UncheckedConsumer> consumer : testClassSpec.allSources) { consumer.accept(sourceSpec); } inputs.put(sourceSpec, Parser.Input.fromString(sourcePath, beforeTrimmed, parser.getCharset(ctx))); } Path relativeTo = testMethodSpec.relativeTo == null ? testClassSpec.relativeTo : testMethodSpec.relativeTo; Iterator> sourceSpecIter = inputs.keySet().iterator(); boolean requirePrintEqualsInput = ctx.getMessage(ExecutionContext.REQUIRE_PRINT_EQUALS_INPUT, true); if (requirePrintEqualsInput) { // this gets checked by the test framework itself a few statements further down ctx.putMessage(ExecutionContext.REQUIRE_PRINT_EQUALS_INPUT, false); } List sourceFiles = parser.parseInputs(inputs.values(), relativeTo, ctx) .collect(Collectors.toList()); assertThat(sourceFiles.size()) .as("Every input should be parsed into a SourceFile.") .isEqualTo(inputs.size()); if (requirePrintEqualsInput) { ctx.putMessage(ExecutionContext.REQUIRE_PRINT_EQUALS_INPUT, true); } for (int i = 0; i < sourceFiles.size(); i++) { SourceFile sourceFile = sourceFiles.get(i); Markers markers = sourceFile.getMarkers(); SourceSpec nextSpec = sourceSpecIter.next(); for (Marker marker : nextSpec.getMarkers()) { markers = markers.setByType(marker); } sourceFile = sourceFile.withMarkers(markers); // Validate before source nextSpec.validateSource.accept(sourceFile, TypeValidation.before(testMethodSpec, testClassSpec)); // Validate that printing the LST yields the same source text // Validate that the LST whitespace do not contain any non-whitespace characters int j = 0; for (Parser.Input input : inputs.values()) { if (j++ == i && !(sourceFile instanceof Quark)) { assertContentEquals( sourceFile, StringUtils.readFully(input.getSource(ctx), parser.getCharset(ctx)), sourceFile.printAll(out.clone()), "When parsing and printing the source code back to text without modifications, " + "the printed source didn't match the original source code. This means there is a bug in the " + "parser implementation itself. Please open an issue to report this, providing a sample of the " + "code that generated this error for" ); try { WhitespaceValidationService service = sourceFile.service(WhitespaceValidationService.class); SourceFile whitespaceValidated = (SourceFile) service.getVisitor().visit(sourceFile, ctx); if(whitespaceValidated != null && whitespaceValidated != sourceFile) { fail("Source file was parsed into an LST that contains non-whitespace characters in its whitespace. " + "This is indicative of a bug in the parser. \n" + whitespaceValidated.printAll()); } } catch (UnsupportedOperationException e) { // Language/parser does not provide whitespace validation and that's OK for now } } } //noinspection unchecked SourceFile mapped = ((UnaryOperator) nextSpec.beforeRecipe).apply(sourceFile); specBySourceFile.put(mapped, nextSpec); } } List beforeSourceFiles = new ArrayList<>(specBySourceFile.keySet()); for (Consumer> beforeRecipe : testClassSpec.beforeRecipes) { beforeRecipe.accept(beforeSourceFiles); } for (Consumer> beforeRecipe : testMethodSpec.beforeRecipes) { beforeRecipe.accept(beforeSourceFiles); } for (SourceFile beforeSourceFile : beforeSourceFiles) { specBySourceFile.put(beforeSourceFile, specBySourceFile.remove(beforeSourceFile)); } List runnableSourceFiles = new ArrayList<>(beforeSourceFiles.size()); for (Map.Entry> sourceFileSpec : specBySourceFile.entrySet()) { if (!sourceFileSpec.getValue().isSkip()) { runnableSourceFiles.add(sourceFileSpec.getKey()); } } ExecutionContext recipeCtx = ctx; if (testMethodSpec.getRecipeExecutionContext() != null) { recipeCtx = testMethodSpec.getRecipeExecutionContext(); } else if (testClassSpec.getRecipeExecutionContext() != null) { recipeCtx = testClassSpec.getRecipeExecutionContext(); } LargeSourceSet lss; if (testMethodSpec.getSourceSet() != null) { lss = testMethodSpec.getSourceSet().apply(runnableSourceFiles); } else if (testClassSpec.getSourceSet() != null) { lss = testClassSpec.getSourceSet().apply(runnableSourceFiles); } else { lss = new LargeSourceSetCheckingExpectedCycles(expectedCyclesThatMakeChanges, runnableSourceFiles); } CursorValidatingExecutionContextView.view(recipeCtx) .setValidateCursorAcyclic(TypeValidation.before(testMethodSpec, testClassSpec) .cursorAcyclic()); RecipeRun recipeRun = recipe.run( lss, recipeCtx, cycles, expectedCyclesThatMakeChanges + 1 ); for (Consumer afterRecipe : testClassSpec.afterRecipes) { afterRecipe.accept(recipeRun); } for (Consumer afterRecipe : testMethodSpec.afterRecipes) { afterRecipe.accept(recipeRun); } Collection> expectedNewSources = Collections.newSetFromMap(new IdentityHashMap<>()); Collection expectedNewResults = Collections.newSetFromMap(new IdentityHashMap<>()); for (SourceSpec sourceSpec : sourceSpecs) { if (sourceSpec.before == null) { expectedNewSources.add(sourceSpec); } } // to prevent a CME inside the next loop expectedNewSources = new CopyOnWriteArrayList<>(expectedNewSources); List allResults = recipeRun.getChangeset().getAllResults(); nextSourceSpec: for (SourceSpec sourceSpec : expectedNewSources) { assertThat(sourceSpec.after).as("Either before or after must be specified in a SourceSpec").isNotNull(); // If the source spec defines a source path, look for a result where there is a new file at that path. if (sourceSpec.getSourcePath() != null) { // If sourceSpec defines a source path, enforce there is a result that has the same source path and // the contents match the expected value. for (Result result : allResults) { if (result.getAfter() != null && sourceSpec.getSourcePath().equals(result.getAfter().getSourcePath())) { expectedNewSources.remove(sourceSpec); expectedNewResults.add(result); assertThat(result.getBefore()) .as("Expected a new file for the source path but there was an existing file already present: " + sourceSpec.getSourcePath()) .isNull(); String actual = result.getAfter().printAll(out.clone()); actual = sourceSpec.noTrim ? actual : actual.trim(); String expected = sourceSpec.noTrim ? sourceSpec.after.apply(actual) : trimIndentPreserveCRLF(sourceSpec.after.apply(actual)); assertThat(actual).as("Unexpected result in \"" + result.getAfter().getSourcePath() + "\"").isEqualTo(expected); continue nextSourceSpec; } } String paths = allResults.stream() .map(it -> { String beforePath = (it.getBefore() == null) ? "null" : it.getBefore().getSourcePath().toString(); String afterPath = (it.getAfter() == null) ? "null" : it.getAfter().getSourcePath().toString(); return " " + beforePath + " -> " + afterPath; }) .collect(Collectors.joining("\n")); fail("Expected a new source file with the source path: " + sourceSpec.getSourcePath() + "\nAll source file paths, before and after recipe run:\n" + paths); } // If the source spec has not defined a source path, look for a result with the exact contents. This logic // first looks for non-remote results. for (Result result : allResults) { if (result.getAfter() != null && !(result.getAfter() instanceof Remote)) { assertThat(sourceSpec.after).as("Either before or after must be specified in a SourceSpec").isNotNull(); String actual = result.getAfter().printAll(out.clone()).trim(); String expected = trimIndentPreserveCRLF(sourceSpec.after.apply(actual)); if (actual.equals(expected)) { expectedNewSources.remove(sourceSpec); expectedNewResults.add(result); //noinspection unchecked ((Consumer) sourceSpec.afterRecipe).accept(result.getAfter()); if (sourceSpec.sourcePath != null) { assertThat(result.getAfter().getSourcePath()) .isEqualTo(sourceSpec.dir.resolve(sourceSpec.sourcePath)); } break; } } } // we tried to avoid it, and now we'll try to match against remotes... for (Result result : allResults) { if (result.getAfter() instanceof Remote) { assertThat(sourceSpec.after).as("Either before or after must be specified in a SourceSpec").isNotNull(); String actual = result.getAfter().printAll(out.clone()); String expected = trimIndentPreserveCRLF(sourceSpec.after.apply(actual)); if (actual.equals(expected)) { expectedNewSources.remove(sourceSpec); //noinspection unchecked ((Consumer) sourceSpec.afterRecipe).accept(result.getAfter()); if (sourceSpec.sourcePath != null) { assertThat(result.getAfter().getSourcePath()) .isEqualTo(sourceSpec.dir.resolve(sourceSpec.sourcePath)); } break; } } } } nextSourceFile: for (Map.Entry> specForSourceFile : specBySourceFile.entrySet()) { SourceSpec sourceSpec = specForSourceFile.getValue(); SourceFile source = specForSourceFile.getKey(); if (source instanceof ParseError) { ParseError parseError = (ParseError) source; if (parseError.getErroneous() != null) { assertContentEquals( parseError, parseError.getText(), parseError.getErroneous().printAll(), "Bug in source parser or printer resulted in the following difference for" ); } } for (Result result : allResults) { if ((result.getBefore() == null && source == null) || (result.getBefore() != null && result.getBefore().getId().equals(source.getId()))) { if (result.getAfter() != null) { String before = result.getBefore() == null ? null : result.getBefore().printAll(out.clone()); String actualAfter = result.getAfter().printAll(out.clone()); String expectedAfter = sourceSpec.after == null ? null : sourceSpec.after.apply(actualAfter); if (expectedAfter != null) { String expected = sourceSpec.noTrim ? expectedAfter : trimIndentPreserveCRLF(expectedAfter); assertContentEquals(result.getAfter(), expected, actualAfter, "Unexpected result in"); sourceSpec.validateSource.accept(result.getAfter(), TypeValidation.after(testMethodSpec, testClassSpec)); } else { boolean isRemote = result.getAfter() instanceof Remote; if (!isRemote && Objects.equals(result.getBefore().getSourcePath(), result.getAfter().getSourcePath()) && Objects.equals(before, actualAfter)) { fail("An empty diff was generated. The recipe incorrectly changed a reference without changing its contents."); } assert result.getBefore() != null; if (isRemote) { // TODO: Verify that the remote URI is correct } else { assertThat(actualAfter) .as("The recipe must not make changes to \"" + result.getBefore().getSourcePath() + "\"") .isEqualTo(before); } } } else { if (sourceSpec.after == null) { // If the source spec was not expecting a change (spec.after == null) but the file has been // deleted, assert failure. assert result.getBefore() != null; fail("The recipe deleted a source file \"" + result.getBefore().getSourcePath() + "\" that was not expected to change"); } else { String expected = sourceSpec.after.apply(null); if (expected != null) { // The spec expected the file to be changed, not deleted. assert result.getBefore() != null; assertThat((String) null) .as("The recipe deleted a source file \"" + result.getBefore().getSourcePath() + "\" but should have changed it instead") .isEqualTo(expected); } } } try { //noinspection unchecked ((Consumer) sourceSpec.afterRecipe).accept(result.getAfter()); } catch (ClassCastException ignored) { // the source file instance type changed, e.g. in FindAndReplace. } continue nextSourceFile; } } // if we get here, there was no result. if (sourceSpec.after != null) { String actual = sourceSpec.noTrim ? source.printAll(out.clone()) : trimIndentPreserveCRLF(source.printAll(out.clone())); String before = sourceSpec.noTrim ? sourceSpec.before : trimIndentPreserveCRLF(sourceSpec.before); String expected = sourceSpec.noTrim ? sourceSpec.after.apply(null) : trimIndentPreserveCRLF(sourceSpec.after.apply(null)); assertThat(expected) .as("To assert that a Recipe makes no change, supply only \"before\" source.") .isNotEqualTo(before); assertThat(actual) .as("The recipe should have made the following change to \"" + source.getSourcePath() + "\"") .isEqualTo(expected); } //noinspection unchecked ((Consumer) sourceSpec.afterRecipe).accept(source); } SoftAssertions newFilesGenerated = new SoftAssertions(); for (SourceSpec expectedNewSource : expectedNewSources) { newFilesGenerated.assertThat(expectedNewSource.after == null ? null : expectedNewSource.after.apply(null)) .as("No new source file was generated that matched.") .isEmpty(); } newFilesGenerated.assertAll(); Map resultToUnexpected = allResults.stream() .collect(Collectors.toMap(result -> result, result -> result.getBefore() == null && !(result.getAfter() instanceof Remote) && !expectedNewResults.contains(result) && testMethodSpec.afterRecipes.isEmpty())); if (resultToUnexpected.values().stream().anyMatch(unexpected -> unexpected)) { String paths = resultToUnexpected.entrySet().stream() .map(it -> { Result result = it.getKey(); assert result.getAfter() != null; String beforePath = (result.getBefore() == null) ? "null" : result.getAfter().getSourcePath().toString(); String afterPath = (result.getAfter() == null) ? "null" : result.getAfter().getSourcePath().toString(); String status = it.getValue() ? "❌️" : "✔"; return " " + beforePath + " | " + afterPath + " | " + status; }) .collect(Collectors.joining("\n")); fail("The recipe generated source files the test did not expect.\n" + "Source file paths before recipe, after recipe, and whether the test expected that result:\n" + " before | after | expected\n" + paths); } } static void assertContentEquals(SourceFile sourceFile, String expected, String actual, String errorMessagePrefix) { try { try (InMemoryDiffEntry diffEntry = new InMemoryDiffEntry( sourceFile.getSourcePath(), sourceFile.getSourcePath(), null, expected, actual, Collections.emptySet() )) { assertThat(actual) .as(errorMessagePrefix + " \"%s\":\n%s", sourceFile.getSourcePath(), diffEntry.getDiff()) .isEqualTo(expected); } } catch (LinkageError e) { // in case JGit fails to load properly assertThat(actual) .as(errorMessagePrefix + " \"%s\"", sourceFile.getSourcePath()) .isEqualTo(expected); } } default void rewriteRun(SourceSpec... sources) { rewriteRun(spec -> { }, sources); } default ExecutionContext defaultExecutionContext(SourceSpec[] sourceSpecs) { InMemoryExecutionContext ctx = new InMemoryExecutionContext(t -> fail("Failed to parse sources or run recipe", t)); ParsingExecutionContextView.view(ctx).setCharset(StandardCharsets.UTF_8); return ctx; } @Override default Iterator> iterator() { return new Iterator>() { @Override public boolean hasNext() { return false; } @Override public SourceSpec next() { throw new UnsupportedOperationException("RewriteTest is not intended to be iterated."); } }; } default void validateRecipeNameAndDescription(Recipe recipe) { if (recipe instanceof CompositeRecipe) { for (Recipe childRecipe : recipe.getRecipeList()) { validateRecipeNameAndDescription(childRecipe); } } else { assertThat(recipe.getDisplayName()).as("%s display name should not end with a period.", recipe.getName()).doesNotEndWith("."); String description = recipe.getDescription(); assertThat(description).as("%s description should not be null or empty", recipe.getName()).isNotEmpty(); if (description.endsWith(")")) { String lastLine = description.substring(description.lastIndexOf('\n') + 1).trim(); assertThat(lastLine).as("%s description not ending with a period should be because the description ends with a markdown list of pure links", recipe.getName()).startsWith("- ["); } else { assertThat(description).as("%s description should end with a period.", recipe.getName()).endsWith("."); } } } default void validateRecipeOptions(Recipe recipe) { for (OptionDescriptor option : recipe.getDescriptor().getOptions()) { assertThat(option.getName()) .as("%s option `name` conflicts with the recipe's name. Please use a different field name for this option.", recipe.getName()) .isNotEqualTo("name"); } } } class RewriteTestUtils { static boolean groupSourceSpecsByParser(List parserBuilders, Map>> sourceSpecsByParser, SourceSpec sourceSpec) { for (Map.Entry>> entry : sourceSpecsByParser.entrySet()) { if (entry.getKey().getSourceFileType().equals(sourceSpec.sourceFileType) && sourceSpec.getParser().getClass().isAssignableFrom(entry.getKey().getClass())) { entry.getValue().add(sourceSpec); return true; } } for (Parser.Builder parser : parserBuilders) { if (parser.getSourceFileType().equals(sourceSpec.sourceFileType)) { sourceSpecsByParser.computeIfAbsent(parser, p -> new ArrayList<>()).add(sourceSpec); return true; } } return false; } } @RequiredArgsConstructor class DelegateSourceFileForDiff implements SourceFile { @Delegate(excludes = PrintAll.class) private final SourceFile delegate; private final String expected; @Override public

String printAll(PrintOutputCapture

out) { out.append(expected); return out.getOut(); } @SuppressWarnings("unused") // Lombok delegate exclude interface PrintAll {

String printAll(PrintOutputCapture

out); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy