繁体   English   中英

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

[英]How do I get Spring MVC to invoke validation in a JUnit test?

我有一个名为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?

谢谢,

马特

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

还有另一种测试控制器的方法,你不直接调用控制器。 相反,您构造并调用控制器映射到的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(需要使用spring里程碑回购):

    <!-- 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版本春季全面实施。

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

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

让它工作的方法是通过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);

基本上你用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+ ,则上述解决方案将无效( 请参阅此链接以获取更多信息 )! 你需要使用这个库 (来自Spring团队,所以听起来不用担心),同时等待它们在下一个Spring版本中集成它。 然后你将不得不做类似的事情

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

还可以看看Rossen Stoyanchev的这些非常有趣的幻灯片 (我们在这里讨论的部分从幻灯片#116开始)!

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

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM