From 4a0edc59cc599815adfaf58e1da9ba0b18957801 Mon Sep 17 00:00:00 2001 From: Sam Brannen <104798+sbrannen@users.noreply.github.com> Date: Wed, 30 Oct 2024 17:31:15 +0100 Subject: [PATCH] Verify support for MockReset for beans within the ApplicationContext MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit This commit verifies that MockReset.before() and MockReset.after() are supported for beans within the ApplicationContext. However, the test class must declare a field annotated with either @⁠MockitoBean or @⁠MockitoSpyBean in order for the MockReset feature to be triggered. See gh-33742 --- ...stenerWithMockitoBeanIntegrationTests.java | 43 ++++ ...outMockitoAnnotationsIntegrationTests.java | 209 ++++++++++++++++++ 2 files changed, 252 insertions(+) create mode 100644 spring-test/src/test/java/org/springframework/test/context/bean/override/mockito/MockitoResetTestExecutionListenerWithMockitoBeanIntegrationTests.java create mode 100644 spring-test/src/test/java/org/springframework/test/context/bean/override/mockito/MockitoResetTestExecutionListenerWithoutMockitoAnnotationsIntegrationTests.java diff --git a/spring-test/src/test/java/org/springframework/test/context/bean/override/mockito/MockitoResetTestExecutionListenerWithMockitoBeanIntegrationTests.java b/spring-test/src/test/java/org/springframework/test/context/bean/override/mockito/MockitoResetTestExecutionListenerWithMockitoBeanIntegrationTests.java new file mode 100644 index 000000000000..496567980492 --- /dev/null +++ b/spring-test/src/test/java/org/springframework/test/context/bean/override/mockito/MockitoResetTestExecutionListenerWithMockitoBeanIntegrationTests.java @@ -0,0 +1,43 @@ +/* + * Copyright 2012-2024 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.springframework.test.context.bean.override.mockito; + +import org.junit.jupiter.api.Test; + +/** + * Integration tests for {@link MockitoResetTestExecutionListener} with a + * {@link MockitoBean @MockitoBean} field. + * + * @author Sam Brannen + * @since 6.2 + * @see MockitoResetTestExecutionListenerWithoutMockitoAnnotationsIntegrationTests + */ +class MockitoResetTestExecutionListenerWithMockitoBeanIntegrationTests + extends MockitoResetTestExecutionListenerWithoutMockitoAnnotationsIntegrationTests { + + // The following mock is not used but is currently required to trigger support for MockReset. + @MockitoBean + StringBuilder unusedVariable; + + + @Test + @Override + void test002() { + super.test002(); + } + +} diff --git a/spring-test/src/test/java/org/springframework/test/context/bean/override/mockito/MockitoResetTestExecutionListenerWithoutMockitoAnnotationsIntegrationTests.java b/spring-test/src/test/java/org/springframework/test/context/bean/override/mockito/MockitoResetTestExecutionListenerWithoutMockitoAnnotationsIntegrationTests.java new file mode 100644 index 000000000000..07375d3312c9 --- /dev/null +++ b/spring-test/src/test/java/org/springframework/test/context/bean/override/mockito/MockitoResetTestExecutionListenerWithoutMockitoAnnotationsIntegrationTests.java @@ -0,0 +1,209 @@ +/* + * Copyright 2012-2024 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.springframework.test.context.bean.override.mockito; + +import org.junit.jupiter.api.Disabled; +import org.junit.jupiter.api.MethodOrderer; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.TestMethodOrder; + +import org.springframework.beans.factory.FactoryBean; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.context.ApplicationContext; +import org.springframework.context.annotation.Bean; +import org.springframework.context.annotation.Configuration; +import org.springframework.context.annotation.Lazy; +import org.springframework.test.context.bean.override.example.ExampleService; +import org.springframework.test.context.junit.jupiter.SpringJUnitConfig; + +import static org.assertj.core.api.Assertions.assertThat; +import static org.mockito.BDDMockito.given; +import static org.mockito.Mockito.mock; + +/** + * Integration tests for {@link MockitoResetTestExecutionListener} without a + * {@link MockitoBean @MockitoBean} or {@link MockitoSpyBean @MockitoSpyBean} field. + * + * @author Phillip Webb + * @author Andy Wilkinson + * @author Sam Brannen + * @since 6.2 + * @see MockitoResetTestExecutionListenerWithMockitoBeanIntegrationTests + */ +@SpringJUnitConfig +@TestMethodOrder(MethodOrderer.MethodName.class) +class MockitoResetTestExecutionListenerWithoutMockitoAnnotationsIntegrationTests { + + @Autowired + ApplicationContext context; + + + @Test + void test001() { + ExampleService nonSingletonFactoryBean = getMock("nonSingletonFactoryBean"); + + given(getMock("none").greeting()).willReturn("none"); + given(getMock("before").greeting()).willReturn("before"); + given(getMock("after").greeting()).willReturn("after"); + given(getMock("singletonFactoryBean").greeting()).willReturn("singletonFactoryBean"); + given(nonSingletonFactoryBean.greeting()).willReturn("nonSingletonFactoryBean"); + + assertThat(getMock("none").greeting()).isEqualTo("none"); + assertThat(getMock("before").greeting()).isEqualTo("before"); + assertThat(getMock("after").greeting()).isEqualTo("after"); + assertThat(getMock("singletonFactoryBean").greeting()).isEqualTo("singletonFactoryBean"); + + // The saved reference should have been mocked. + assertThat(nonSingletonFactoryBean.greeting()).isEqualTo("nonSingletonFactoryBean"); + // A new reference should have not been mocked. + assertThat(getMock("nonSingletonFactoryBean").greeting()).isNull(); + + // getMock("nonSingletonFactoryBean") has been invoked twice in this method. + assertThat(context.getBean(NonSingletonFactoryBean.class).getObjectInvocations).isEqualTo(2); + } + + @Disabled("MockReset is currently only honored if @MockitoBean or @MockitoSpyBean is used") + @Test + void test002() { + // Should not have been reset. + assertThat(getMock("none").greeting()).isEqualTo("none"); + + // Should have been reset. + assertThat(getMock("before").greeting()).isNull(); + assertThat(getMock("after").greeting()).isNull(); + assertThat(getMock("singletonFactoryBean").greeting()).isNull(); + + // A non-singleton FactoryBean always creates a new mock instance. Thus, + // resetting is irrelevant, and the greeting should be null. + assertThat(getMock("nonSingletonFactoryBean").greeting()).isNull(); + + // getMock("nonSingletonFactoryBean") has been invoked twice in test001() + // and once in this method. + assertThat(context.getBean(NonSingletonFactoryBean.class).getObjectInvocations).isEqualTo(3); + } + + private ExampleService getMock(String name) { + return context.getBean(name, ExampleService.class); + } + + + @Configuration(proxyBeanMethods = false) + static class Config { + + @Bean + ExampleService none() { + return mock(ExampleService.class); + } + + @Bean + ExampleService before() { + return mock(ExampleService.class, MockReset.before()); + } + + @Bean + ExampleService after() { + return mock(ExampleService.class, MockReset.after()); + } + + @Bean + @Lazy + ExampleService fail() { + // Spring Boot gh-5870 + throw new RuntimeException(); + } + + @Bean + BrokenFactoryBean brokenFactoryBean() { + // Spring Boot gh-7270 + return new BrokenFactoryBean(); + } + + @Bean + WorkingFactoryBean singletonFactoryBean() { + return new WorkingFactoryBean(); + } + + @Bean + NonSingletonFactoryBean nonSingletonFactoryBean() { + return new NonSingletonFactoryBean(); + } + + } + + static class BrokenFactoryBean implements FactoryBean { + + @Override + public String getObject() { + throw new IllegalStateException(); + } + + @Override + public Class getObjectType() { + return String.class; + } + + @Override + public boolean isSingleton() { + return true; + } + + } + + static class WorkingFactoryBean implements FactoryBean { + + private final ExampleService service = mock(ExampleService.class, MockReset.before()); + + @Override + public ExampleService getObject() { + return this.service; + } + + @Override + public Class getObjectType() { + return ExampleService.class; + } + + @Override + public boolean isSingleton() { + return true; + } + + } + + static class NonSingletonFactoryBean implements FactoryBean { + + private int getObjectInvocations = 0; + + @Override + public ExampleService getObject() { + this.getObjectInvocations++; + return mock(ExampleService.class, MockReset.before()); + } + + @Override + public Class getObjectType() { + return ExampleService.class; + } + + @Override + public boolean isSingleton() { + return false; + } + + } + +}