Skip to content

Commit

Permalink
Refactor Mockito.when on static (non mock) to try with resource (#601)
Browse files Browse the repository at this point in the history
* WIP

* Format and suggestion

* Add recipe to mockito recipes
Add missing imports
Fill out name and desc

* Update test with TypeValidation

* Format and refactor to remove errors

* Reduce and flatten ahead of further changes

* Show problematic case with new unit test

* Generate a non-conflicting variable name of the right type

* Switch to `ListUtils.flatMap` without `mock` field in test

* Apply suggestions from code review

Co-authored-by: github-actions[bot] <41898282+github-actions[bot]@users.noreply.github.com>

* Make recursive in case multiple occurrences exist

* Return immediately

* Reduce nesting and clarify intent through method rename

* Only add preceding statements before try, not all

* Shorten qualified class refs & enable method type validation

---------

Co-authored-by: Tim te Beek <[email protected]>
Co-authored-by: github-actions[bot] <41898282+github-actions[bot]@users.noreply.github.com>
  • Loading branch information
3 people authored Sep 21, 2024
1 parent 5be8a6f commit 4367149
Show file tree
Hide file tree
Showing 3 changed files with 266 additions and 0 deletions.
Original file line number Diff line number Diff line change
@@ -0,0 +1,115 @@
/*
* Copyright 2024 the original author or authors.
* <p>
* 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
* <p>
* https://www.apache.org/licenses/LICENSE-2.0
* <p>
* 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.java.testing.mockito;

import org.openrewrite.ExecutionContext;
import org.openrewrite.Preconditions;
import org.openrewrite.Recipe;
import org.openrewrite.TreeVisitor;
import org.openrewrite.internal.ListUtils;
import org.openrewrite.java.JavaIsoVisitor;
import org.openrewrite.java.JavaTemplate;
import org.openrewrite.java.MethodMatcher;
import org.openrewrite.java.VariableNameUtils;
import org.openrewrite.java.search.UsesMethod;
import org.openrewrite.java.tree.Flag;
import org.openrewrite.java.tree.J;
import org.openrewrite.java.tree.Statement;

import java.util.Collections;
import java.util.List;
import java.util.concurrent.atomic.AtomicBoolean;

public class MockitoWhenOnStaticToMockStatic extends Recipe {

private static final MethodMatcher MOCKITO_WHEN = new MethodMatcher("org.mockito.Mockito when(..)");

@Override
public String getDisplayName() {
return "Replace `Mockito.when` on static (non mock) with try-with-resource with MockedStatic";
}

@Override
public String getDescription() {
return "Replace `Mockito.when` on static (non mock) with try-with-resource with MockedStatic as Mockito4 no longer allows this.";
}

@Override
public TreeVisitor<?, ExecutionContext> getVisitor() {
return Preconditions.check(new UsesMethod<>(MOCKITO_WHEN), new JavaIsoVisitor<ExecutionContext>() {
@Override
public J.MethodDeclaration visitMethodDeclaration(J.MethodDeclaration method, ExecutionContext ctx) {
J.MethodDeclaration m = super.visitMethodDeclaration(method, ctx);
if (m.getBody() == null) {
return m;
}

List<Statement> newStatements = maybeWrapStatementsInTryWithResourcesMockedStatic(m, m.getBody().getStatements());
return maybeAutoFormat(m, m.withBody(m.getBody().withStatements(newStatements)), ctx);
}

private List<Statement> maybeWrapStatementsInTryWithResourcesMockedStatic(J.MethodDeclaration m, List<Statement> remainingStatements) {
AtomicBoolean restInTry = new AtomicBoolean(false);
return ListUtils.flatMap(remainingStatements, (index, statement) -> {
if (restInTry.get()) {
// Rest of the statements have ended up in the try block
return Collections.emptyList();
}

if (statement instanceof J.MethodInvocation &&
MOCKITO_WHEN.matches(((J.MethodInvocation) statement).getSelect())) {
J.MethodInvocation when = (J.MethodInvocation) ((J.MethodInvocation) statement).getSelect();
if (when != null && when.getArguments().get(0) instanceof J.MethodInvocation) {
J.MethodInvocation whenArg = (J.MethodInvocation) when.getArguments().get(0);
if (whenArg.getMethodType() != null && whenArg.getMethodType().hasFlags(Flag.Static)) {
J.Identifier clazz = (J.Identifier) whenArg.getSelect();
if (clazz != null && clazz.getType() != null) {
String mockName = VariableNameUtils.generateVariableName("mock" + clazz.getSimpleName(), updateCursor(m), VariableNameUtils.GenerationStrategy.INCREMENT_NUMBER);
maybeAddImport("org.mockito.MockedStatic", false);
maybeAddImport("org.mockito.Mockito", "mockStatic");
String template = String.format(
"try(MockedStatic<%1$s> %2$s = mockStatic(%1$s.class)) {\n" +
" %2$s.when(#{any()}).thenReturn(#{any()});\n" +
"}", clazz.getSimpleName(), mockName);
J.Try try_ = (J.Try) ((J.MethodDeclaration) JavaTemplate.builder(template)
.contextSensitive()
.imports("org.mockito.MockedStatic")
.staticImports("org.mockito.Mockito.mockStatic")
.build()
.apply(getCursor(), m.getCoordinates().replaceBody(),
whenArg, ((J.MethodInvocation) statement).getArguments().get(0)))
.getBody().getStatements().get(0);

restInTry.set(true);

List<Statement> precedingStatements = remainingStatements.subList(0, index);
return try_.withBody(try_.getBody().withStatements(ListUtils.concatAll(
try_.getBody().getStatements(),
maybeWrapStatementsInTryWithResourcesMockedStatic(
m.withBody(m.getBody().withStatements(ListUtils.concat(precedingStatements, try_))),
remainingStatements.subList(index + 1, remainingStatements.size())
))))
.withPrefix(statement.getPrefix());
}
}
}
}
return statement;
});
}
});
}
}
1 change: 1 addition & 0 deletions src/main/resources/META-INF/rewrite/mockito.yml
Original file line number Diff line number Diff line change
Expand Up @@ -59,6 +59,7 @@ tags:
- mockito
recipeList:
- org.openrewrite.java.testing.mockito.Mockito1to3Migration
- org.openrewrite.java.testing.mockito.MockitoWhenOnStaticToMockStatic
- org.openrewrite.java.dependencies.UpgradeDependencyVersion:
groupId: org.mockito
artifactId: "*"
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,150 @@
/*
* Copyright 2024 the original author or authors.
* <p>
* 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
* <p>
* https://www.apache.org/licenses/LICENSE-2.0
* <p>
* 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.java.testing.mockito;

import org.junit.jupiter.api.Test;
import org.openrewrite.DocumentExample;
import org.openrewrite.InMemoryExecutionContext;
import org.openrewrite.java.JavaParser;
import org.openrewrite.test.RecipeSpec;
import org.openrewrite.test.RewriteTest;
import org.openrewrite.test.SourceSpec;
import org.openrewrite.test.TypeValidation;

import static org.openrewrite.java.Assertions.java;

class MockitoWhenOnStaticToMockStaticTest implements RewriteTest {

@Override
public void defaults(RecipeSpec spec) {
spec.recipe(new MockitoWhenOnStaticToMockStatic())
.parser(JavaParser.fromJavaVersion()
.classpathFromResources(new InMemoryExecutionContext(),
"junit-4.13",
"mockito-core-3.12",
"mockito-junit-jupiter-3.12"
));
}

@DocumentExample
@Test
void shouldRefactorMockito_When() {
//language=java
rewriteRun(
spec -> spec.afterTypeValidationOptions(TypeValidation.builder().identifiers(false).build()),
java(
"""
package com.foo;
public class A {
public static Integer getNumber() {
return 42;
}
}
""",
SourceSpec::skip
),
java(
"""
import com.foo.A;
import static org.junit.Assert.assertEquals;
import static org.mockito.Mockito.*;
class Test {
void test() {
when(A.getNumber()).thenReturn(-1);
assertEquals(A.getNumber(), -1);
}
}
""",
"""
import com.foo.A;
import org.mockito.MockedStatic;
import static org.junit.Assert.assertEquals;
import static org.mockito.Mockito.*;
class Test {
void test() {
try (MockedStatic<A> mockA = mockStatic(A.class)) {
mockA.when(A.getNumber()).thenReturn(-1);
assertEquals(A.getNumber(), -1);
}
}
}
"""
)
);
}

@Test
void shouldHandleMultipleStaticMocks() {
//language=java
rewriteRun(
spec -> spec.afterTypeValidationOptions(TypeValidation.builder().identifiers(false).build()),
java(
"""
package com.foo;
public class A {
public static Integer getNumber() {
return 42;
}
}
""",
SourceSpec::skip
),
java(
"""
import com.foo.A;
import static org.junit.Assert.assertEquals;
import static org.mockito.Mockito.*;
class Test {
void test() {
when(A.getNumber()).thenReturn(-1);
assertEquals(A.getNumber(), -1);
when(A.getNumber()).thenReturn(-2);
assertEquals(A.getNumber(), -2);
}
}
""",
"""
import com.foo.A;
import org.mockito.MockedStatic;
import static org.junit.Assert.assertEquals;
import static org.mockito.Mockito.*;
class Test {
void test() {
try (MockedStatic<A> mockA = mockStatic(A.class)) {
mockA.when(A.getNumber()).thenReturn(-1);
assertEquals(A.getNumber(), -1);
try (MockedStatic<A> mockA1 = mockStatic(A.class)) {
mockA1.when(A.getNumber()).thenReturn(-2);
assertEquals(A.getNumber(), -2);
}
}
}
}
"""
)
);
}
}

0 comments on commit 4367149

Please sign in to comment.