在测试前后使 ApplicationContext 变脏 class

Make ApplicationContext dirty before and after test class

我的 Spring 集成测试中有一个特定的 class(比方说 MyTest),它在 Spring 组件上使用 PowerMock @PrepareForTest 注释:@PrepareForTest(MyComponent.class)。这意味着 PowerMock 将加载此 class 并进行一些修改。问题是,我的 @ContextConfiguration 定义在由 MyTest 扩展的 superclass 上,而 ApplicationContext 在不同的测试 class 之间缓存。现在,如果 MyTest 首先是 运行,它将具有 MyComponent 的正确 PowerMock 版本,但如果不是 - 测试将失败,因为上下文将为另一个测试加载(没有 @准备测试)。

所以我想做的是在 MyTest 之前重新加载我的上下文。我可以通过

@DirtiesContext(classMode = DirtiesContext.ClassMode.BEFORE_CLASS)

但是如果我还想在完成此测试后重新加载上下文怎么办?所以我将在没有 PowerMock 修改的情况下再次获得干净的 MyComponent。有没有办法同时执行 BEFORE_CLASSAFTER_CLASS

现在我用以下 hack 做到了:

@FixMethodOrder(MethodSorters.NAME_ASCENDING)

在 MyTest 上然后

/**
 * Stub test to reload ApplicationContext before execution of real test methods of this class.
 */
@DirtiesContext(methodMode = DirtiesContext.MethodMode.BEFORE_METHOD)
@Test
public void aa() {
}

/**
 * Stub test to reload ApplicationContext after execution of real test methods of this class.
 */
@DirtiesContext(methodMode = DirtiesContext.MethodMode.AFTER_METHOD)
@Test
public void zz() {
}

我想知道是否有更好的方法来做到这一点?

作为附带问题,是否可以仅重新加载某些 bean 而不是完整的上下文?

Is there a way to do both BEFORE_CLASS and AFTER_CLASS?

不,不幸的是 @DirtiesContext 不支持。

但是,您真正想说的是,您想要一个新的 ApplicationContext 用于 MyTest,它与父测试 class 的上下文相同,但只有一样长作为 MyTest。并且...您不想影响为父测试缓存的上下文 class.

考虑到这一点,下面的技巧应该可以做到。

@RunWith(SpringJUnit4ClassRunner.class)
// Inherit config from parent and combine with local 
// static Config class to create a new context
@ContextConfiguration
@DirtiesContext
public class MyTest extends BaseTests {

    @Configuration
    static class Config {
        // No need to define any actual @Bean methods.
        // We only need to add an additional @Configuration
        // class so that we get a new ApplicationContext.
    }
}

替代@DirtiesContext

如果你想在测试之前之后都弄脏上下文 class,你可以实现自定义TestExecutionListener 正是这样做的。例如,以下内容可以解决问题。

import org.springframework.core.Ordered;
import org.springframework.test.annotation.DirtiesContext.HierarchyMode;
import org.springframework.test.context.TestContext;
import org.springframework.test.context.support.AbstractTestExecutionListener;

public class DirtyContextBeforeAndAfterClassTestExecutionListener
        extends AbstractTestExecutionListener {

    @Override
    public int getOrder() {
        return Ordered.HIGHEST_PRECEDENCE;
    }

    @Override
    public void beforeTestClass(TestContext testContext) throws Exception {
        testContext.markApplicationContextDirty(HierarchyMode.EXHAUSTIVE);
    }

    @Override
    public void afterTestClass(TestContext testContext) throws Exception {
        testContext.markApplicationContextDirty(HierarchyMode.EXHAUSTIVE);
    }

}

然后您可以在 MyTest 中使用自定义侦听器,如下所示。

import org.springframework.test.context.TestExecutionListeners;
import org.springframework.test.context.TestExecutionListeners.MergeMode;

@TestExecutionListeners(
    listeners = DirtyContextBeforeAndAfterClassTestExecutionListener.class, 
    mergeMode = MergeMode.MERGE_WITH_DEFAULTS
)
public class MyTest extends BaseTest { /* ... */ }

As a side question, is it possible to reload only certain bean and not full context?

不,那也不可能。

此致,

Sam(Spring TestContext Framework 的作者)