From java
OpenRewrite recipe test maintenance. Use when fixing test failures, import ordering issues, type validation problems, IDE warnings, or writing comprehensive recipe tests.
npx claudepluginhub motlin/claude-code-plugins --plugin javaThis skill uses the workspace's default tool permissions.
This skill provides guidelines for developing OpenRewrite recipes and maintaining their tests, with a focus on import ordering issues.
Creates isolated Git worktrees for feature branches with prioritized directory selection, gitignore safety checks, auto project setup for Node/Python/Rust/Go, and baseline verification.
Executes implementation plans in current session by dispatching fresh subagents per independent task, with two-stage reviews: spec compliance then code quality.
Dispatches parallel agents to independently tackle 2+ tasks like separate test failures or subsystems without shared state or dependencies.
This skill provides guidelines for developing OpenRewrite recipes and maintaining their tests, with a focus on import ordering issues.
๐ง OpenRewrite recipe tests often fail due to import order differences, not actual transformation issues.
OpenRewrite recipe tests fail with diffs showing only import order differences:
-import java.util.List;
-import org.assertj.core.api.Assertions;
+import org.assertj.core.api.Assertions;
+import java.util.List;
OpenRewrite manages imports automatically based on:
Ensure your JavaTemplate is properly configured:
JavaTemplate template = JavaTemplate
.builder("Your.template.code()")
.imports(
"org.assertj.core.api.Assertions",
"org.eclipse.collections.impl.utility.Iterate"
)
.contextSensitive() // Important for proper context handling
.javaParser(JavaParser.fromJavaVersion()
.classpath("assertj-core", "eclipse-collections", "eclipse-collections-api")
)
.build();
Don't forget to call:
maybeAddImport("org.assertj.core.api.Assertions");
maybeAddImport("org.eclipse.collections.impl.utility.Iterate");
maybeRemoveImport("old.package.OldClass");
Accept the actual import order that OpenRewrite produces:
โ Instead of forcing a specific order:
// DON'T expect a specific order you want
"import java.util.List;\n" +
"import org.assertj.core.api.Assertions;\n"
โ Use the actual order OpenRewrite produces:
// DO accept the order OpenRewrite generates
"import org.assertj.core.api.Assertions;\n" +
"import org.eclipse.collections.impl.utility.Iterate;\n" +
"\n" +
"import java.util.List;\n"
OpenRewrite typically orders imports as:
java.*, javax.*)The ~~> prefix in test expectations is not standard in all codebases. It's used in some OpenRewrite projects to indicate "ignore everything before this line" but isn't recognized in all contexts. If you see it failing, remove it and use exact matching instead.
@Test
void replacesVerifyWithAssertJ() {
rewriteRun(
java(
// Input
"""
import org.eclipse.collections.impl.test.Verify;
import java.util.List;
class Test {
void test() {
List<String> list = List.of("a", "b", "c");
Verify.assertCount(2, list, each -> each.length() > 0);
}
}
""",
// Expected output - use actual order from test failure
"""
import org.assertj.core.api.Assertions;
import org.eclipse.collections.impl.utility.Iterate;
import java.util.List;
class Test {
void test() {
List<String> list = List.of("a", "b", "c");
Assertions.assertThat(Iterate.count(list, each -> each.length() > 0)).isEqualTo(2);
}
}
"""
)
);
}
For tests involving custom types with incomplete type information, disable type validation as a last resort. Prefer specifying types that exist:
@Test
void withCustomType() {
rewriteRun(
spec -> spec.typeValidationOptions(TypeValidation.none()),
java(
// test code
)
);
}
When testing code with intentional issues (that the recipe will fix), suppress IDE warnings:
// Single test method
@SuppressWarnings("RedundantCast")
@Test
void testRedundantCast() { ... }
// Multiple tests with same warning - move to class level
@SuppressWarnings({"ConstantConditions", "RedundantCast"})
class MyRecipeTest implements RewriteTest { ... }
Common suppressions: "RedundantCast", "ConstantConditions", "unused", "unchecked"
Add //language=java before string templates to enable IDE syntax highlighting.
When using java("before", "after") with no customization, place the comment before java:
//language=java
java(
"""
public class Before { }
""",
"""
public class After { }
"""
)
When there's customization or multiple java() calls, place comments on individual strings:
spec -> spec.typeValidationOptions(TypeValidation.none()),
//language=java
java(
"""
public class Test { }
"""
)
Do NOT add //language=java to JavaTemplate strings containing parameters like #{any()} or #{} โ these aren't valid Java and will cause IDE errors.
Ensure comprehensive coverage including:
Maven POM files should follow a consistent dependency ordering structure. See the pom-ordering skill for detailed guidelines.