使用自动装配的 MockHttpServletRequest 进行的多次测试不起作用?

hol*_*s83 2 testng spring spring-test

@Autowired MockHttpServletRequest在一些 Spring 测试中使用了 。TestNG用作测试框架。如果我在课堂上只有一种测试方法,那么效果很好。但是,如果有多个测试方法,则只有第一个运行测试使用我的 MockHttpServletRequest。让我用一个例子来说明:

@WebAppConfiguration
@ContextConfiguration({"classpath:applicationContext.xml"})
public class FooTest extends AbstractTestNGSpringContextTests {

    @Autowired
    private MockHttpServletRequest servletRequest;

    @Test
    public void test1() {
        assertEquals(((ServletRequestAttributes)RequestContextHolder.getRequestAttributes()).getRequest(), servletRequest);
    }

    @Test
    public void test2() {
        assertEquals(((ServletRequestAttributes)RequestContextHolder.getRequestAttributes()).getRequest(), servletRequest);
    }

}
Run Code Online (Sandbox Code Playgroud)

在此示例中,test1()通过了,但test2()失败了!如果单独运行测试方法,它们都会通过。如果一起运行,为什么一项测试会失败?

我尝试深入研究代码,在测试方法运行后似乎会重置请求属性,但我没有找到将其关闭的方法。我的 Spring 版本是 3.2.8.RELEASE。

Sam*_*nen 5

更新:这已在 Spring Framework 3.2.9、4.0.4 和 4.1 中修复。有关详细信息,请参阅SPR-11626


好吧,我的朋友...您发现了Spring TestContext Framework中的一个错误。

出现此行为的原因是ServletTestExecutionListener在每个测试方法之后重置请求属性,但DependencyInjectionTestExecutionListener不会在每个测试方法之前重新注入依赖项(默认情况下)。当执行第二个测试方法时,该servletRequest字段仍然引用MockHttpServletRequest为前一个测试方法创建的字段;然而,为每个测试方法ServletTestExecutionListener创建一个新实例MockHttpServletRequest并将其设置在请求属性中。因此,注入的请求和存储RequestContextHolder在 TestNG 中的第一个测试方法中存储的请求仅相同。

由于我是这段代码的作者,我必须亲自道歉,但是......我会确保它尽快得到修复。有关修复状态的详细信息,请参阅SPR-11626 。;)

注意:此错误仅适用于 TestNG 测试;这不适用于JUnit 测试。

作为解决方法,您可以使用 注释受影响的测试方法@DirtiesContext(或使用 注释您的测试类@DirtiesContext(classMode = ClassMode.AFTER_EACH_TEST_METHOD))。这将使您的测试按您的预期通过。

使用@DirtiesContext将使 SpringApplicationContext在每个测试方法之后关闭你的测试,这可能会对你的测试速度产生负面影响;然而,从 Spring 3.2.8 和 4.0.3 开始,这是唯一的非自定义解决方案。

话虽如此,以下是一种更有效的解决方法。只需在您的项目中定义此自定义即可TestExecutionListener

public class AlwaysReinjectDependenciesTestExecutionListener extends AbstractTestExecutionListener {

    public void afterTestMethod(TestContext testContext) throws Exception {
        testContext.setAttribute(DependencyInjectionTestExecutionListener.REINJECT_DEPENDENCIES_ATTRIBUTE, Boolean.TRUE);
    }

}
Run Code Online (Sandbox Code Playgroud)

然后像这样注释你的测试类:

@TestExecutionListeners(AlwaysReinjectDependenciesTestExecutionListener.class)
Run Code Online (Sandbox Code Playgroud)

这应该可以解决所有问题让您的测试套件快速运行。

问候,

山姆