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

时间:2012-09-06 20:52:38

标签: java spring spring-mvc junit bean-validation

我有一个名为Browser的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());
}

这是我的Controller的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?

谢谢,

马特

3 个答案:

答案 0 :(得分:5)

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

还有另一种测试控制器的方法,你不直接调用控制器。相反,您构造并调用控制器映射到的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版春季全面实施。

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

答案 1 :(得分:2)

在将请求绑定到方法参数的过程中,在调用的控制器方法之前调用验证器。因为在这种情况下,您直接调用控制器方法,所以绕过了绑定和验证步骤。

让它工作的方法是通过Spring MVC堆栈调用控制器 - 有几种方法可以做到这一点,我觉得最好的方法是使用spring-test-mvc提供一个通过堆栈调用的好机制。

调用堆栈的另一种方法是以这种方式将HandlerAdapter注入到测试中:

@Autowired
private RequestMappingHandlerAdapter handlerAdapter;

然后在测试中:

MockHttpServletRequest request = new MockHttpServletRequest("POST","/browser/create");
MockHttpServletResponse response = new MockHttpServletResponse();
httpRequest.addParameter(.... );//whatever is required to create Browser..
ModelAndView modelAndView = handlerAdapter.handle(httpRequest, response, handler);

答案 2 :(得分:2)

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

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

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

如果您使用的是Spring 3.0.x或更低版本,则需要执行

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

让它发挥作用。

如果您使用的是Spring 3.1 + ,则上述解决方案无效(see this link for more info)!你需要使用this library(来自Spring团队,所以听起来不用担心),等待他们将它集成到下一个Spring版本中。 然后你将不得不做类似

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

另请参阅Rossen Stoyanchev的这些interesting slides(我们在这里讨论的部分从第116页幻灯片开始)!

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