如何让 Spring MVC 在 JUnit 测试中调用验证?

我有一个名为浏览器的POJO,我用Hibernate Validator注释进行了注释。

import org.hibernate.validator.constraints.NotEmpty;

public class Browser {

    @NotEmpty
    private String userAgent;
    @NotEmpty
    private String browserName;

...

}

我已经编写了以下单元测试,该测试尝试验证我的 Controller 方法是否捕获验证错误。

@Test
public void testInvalidData() throws Exception {
    Browser browser = new Browser("opera", null);
    MockHttpServletRequest request = new MockHttpServletRequest();

    BindingResult errors = new DataBinder(browser).getBindingResult();
    // controller is initialized in @Before method
    controller.add(browser, errors, request);
    assertEquals(1, errors.getErrorCount());
}

这是我的控制器的 add() 方法:

@RequestMapping(value = "/browser/create", method = RequestMethod.POST)
public String add(@Valid Browser browser, BindingResult result, HttpServletRequest request) throws Exception {
    if (result.hasErrors()) {
        request.setAttribute("errorMessage", result.getAllErrors());
        return VIEW_NAME;
    }

    browserManager.save(browser);

    request.getSession(false).setAttribute("successMessage",
            String.format("Browser %s added successfully.", browser.getUserAgent()));

    return "redirect:/" + VIEW_NAME;
}

我遇到的问题是,结果永远不会有错误,所以就像@Valid没有被识别一样。我尝试将以下内容添加到我的测试类中,但它不能解决问题。

@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration({"file:path-to/WEB-INF/spring-mvc-servlet.xml"})

有谁知道在使用JUnit进行测试时,我如何获得@Valid被识别(和验证)?

谢谢

马 特


答案 1

验证是在调用控制器之前完成的,因此测试不会调用此验证。

还有另一种测试控制器的方法,其中不直接调用控制器。相反,您可以构造并调用控制器映射到的 URL。以下是如何执行此操作的一个很好的例子:http://rstoyanchev.github.com/spring-31-and-mvc-test/#1

@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(loader=WebContextLoader.class, locations = {"classpath:/META-INF/spring/applicationContext.xml", "classpath:/META-INF/spring/applicationContext-test-override.xml", "file:src/main/webapp/WEB-INF/spring/webmvc-config.xml"})
public class MyControllerTest {
@Autowired
WebApplicationContext wac;
MockMvc mockMvc;

@Before
public void setup() {
    this.mockMvc = MockMvcBuilders.webApplicationContextSetup(this.wac).build();
}

@Test
@Transactional
public void testMyController() throws Exception {
    this.mockMvc.perform(get("/mycontroller/add?param=1").accept(MediaType.TEXT_HTML))
    .andExpect(status().isOk())
    .andExpect(model().attribute("date_format", "M/d/yy h:mm a"))
    .andExpect(model().attribute("myvalue", notNullValue()))
    .andExpect(model().attribute("myvalue", hasSize(2)))
    .andDo(print());
}
}

POM(需要使用春季里程碑回购):

    <!-- required for spring-test-mvc -->
    <repository>
        <id>spring-maven-milestone</id>
        <name>Spring Maven Milestone Repository</name>
        <url>http://maven.springframework.org/milestone</url>
    </repository>
...
    <dependency>
        <groupId>org.springframework</groupId>
        <artifactId>spring-test-mvc</artifactId>
        <version>1.0.0.M1</version>
        <scope>test</scope>
    </dependency>

注意:spring-mvc-test lib 尚未准备好生产。在实施中存在一些差距。我认为它计划在春季3.2中全面实施。

这种方法是一个好主意,因为它可以全面测试您的控制器。它很容易弄乱你的控制器映射,所以这些确实需要进行单元测试。


答案 2

基本上,您用 实例化了一个 POJO,然后调用其方法。只是简单的Java和一个简单的对象,没有任何上下文:@Valid没有被考虑在内。this.controller = new MyController()this.controller.add(...)

@ContextConfiguration只会加载可能的bean,可能的自定义验证器等,但它不会像处理@Valid那样神奇。

您需要的是模拟对控制器方法的请求。完全模拟它,包括验证。你离这样做已经不远了,因为你使用了一些Spring测试工具(来实例化MockHttpServletRequest)。add

如果您使用Spring 3.0.x或更低,则需要做

new AnnotationMethodHandlerAdapter()
      .handle(request, new MockHttpServletResponse(), this.controller);

使其正常工作。

如果您使用Spring 3.1 +,上述解决方案将不起作用(请参阅此链接以获取更多信息)!您将需要使用此库(来自Spring团队,因此不必担心它的声音),同时等待他们将其集成到下一个Spring版本中。然后,您将不得不做这样的事情

myMockController = MockMvcBuilders.standaloneSetup(new MyController()).build();
myMockController.perform(get("/browser/create")).andExpect(...);

也看看罗森·斯托扬切夫(Rossen Stoyanchev)的这些非常有趣的幻灯片(我们在这里谈论的部分从幻灯片#116开始)!

注意:我不会进入关于这种测试是否被视为单元测试或集成测试的辩论。有人会说,这是我们在这里做的集成测试,因为我们模拟了请求的完整路径。但另一方面,你仍然可以用来自Mockito的@Mock注释来模拟你的控制器(或者用任何其他模拟框架做类似的事情),所以其他人会说你可以将测试的范围缩小到几乎纯粹的“单元测试”。当然,你可以使用普通的旧Java +模拟框架来替代和纯粹的单元测试你的控制器,但在这种情况下,这将不允许您测试@Valid验证。做出您的选择!:)


推荐