Тестирование Spring @MVC аннотации - PullRequest
9 голосов
/ 23 февраля 2010

На днях я столкнулся с проблемой, когда аннотация @Valid была случайно удалена из класса контроллера. К сожалению, это не сломало ни одного из наших тестов. Ни один из наших модульных тестов не использует путь Spring AnnotationMethodHandlerAdapter. Мы просто тестируем наши классы контроллеров напрямую.

Как я могу написать модульный или интеграционный тест, который не будет работать правильно, если мои аннотации @MVC неверны? Есть ли способ попросить Spring найти и использовать соответствующий контроллер с MockHttpServlet или чем-то еще?

Ответы [ 3 ]

13 голосов
/ 29 июля 2010

Я пишу интеграционные тесты для такого рода вещей. Допустим, у вас есть компонент с аннотациями проверки:

public class MyForm {
    @NotNull
    private Long myNumber;

    ...
}

и контроллер, который обрабатывает отправку

@Controller
@RequestMapping("/simple-form")
public class MyController {
    private final static String FORM_VIEW = null;

    @RequestMapping(method = RequestMethod.POST)
    public String processFormSubmission(@Valid MyForm myForm,
            BindingResult result) {
        if (result.hasErrors()) {
            return FORM_VIEW;
        }
        // process the form
        return "success-view";
    }
}

и вы хотите проверить правильность подключения аннотаций @Valid и @NotNull:

@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration({"file:web/WEB-INF/application-context.xml",
    "file:web/WEB-INF/dispatcher-servlet.xml"})
public class MyControllerIntegrationTest {

    @Inject
    private ApplicationContext applicationContext;

    private MockHttpServletRequest request;
    private MockHttpServletResponse response;
    private HandlerAdapter handlerAdapter;

    @Before
    public void setUp() throws Exception {
        this.request = new MockHttpServletRequest();
        this.response = new MockHttpServletResponse();

        this.handlerAdapter = applicationContext.getBean(HandlerAdapter.class);
    }

    ModelAndView handle(HttpServletRequest request, HttpServletResponse response)
            throws Exception {
        final HandlerMapping handlerMapping = applicationContext.getBean(HandlerMapping.class);
        final HandlerExecutionChain handler = handlerMapping.getHandler(request);
        assertNotNull("No handler found for request, check you request mapping", handler);

        final Object controller = handler.getHandler();
        // if you want to override any injected attributes do it here

        final HandlerInterceptor[] interceptors =
            handlerMapping.getHandler(request).getInterceptors();
        for (HandlerInterceptor interceptor : interceptors) {
            final boolean carryOn = interceptor.preHandle(request, response, controller);
            if (!carryOn) {
                return null;
            }
        }

        final ModelAndView mav = handlerAdapter.handle(request, response, controller);
        return mav;
    }

    @Test
    public void testProcessFormSubmission() throws Exception {
        request.setMethod("POST");
        request.setRequestURI("/simple-form");
        request.setParameter("myNumber", "");

        final ModelAndView mav = handle(request, response);
        // test we're returned back to the form
        assertViewName(mav, "simple-form");
        // make assertions on the errors
        final BindingResult errors = assertAndReturnModelAttributeOfType(mav, 
                "org.springframework.validation.BindingResult.myForm", 
                BindingResult.class);
        assertEquals(1, errors.getErrorCount());
        assertEquals("", errors.getFieldValue("myNumber"));        
    }

См. Мое сообщение в блоге о тестировании интеграции аннотаций Spring MVC

3 голосов
/ 23 февраля 2010

Конечно. Нет причины, по которой ваш тест не может создать свою собственную DispatcherServlet, внедрить в нее различные элементы, которые он будет иметь в контейнере (например, ServletContext), включая расположение файла определения контекста.

Spring поставляется с различными сервлет-классами MockXYZ для этой цели, включая MockServletContext, MockHttpServletRequest и MockHttpServletResponse. На самом деле они не "насмешливые" объекты в обычном смысле этого слова, они больше похожи на тупых тупиков, но они делают свою работу.

Тестовый контекст сервлета будет содержать обычные bean-компоненты, связанные с MVC, плюс ваши bean-компоненты для тестирования. После инициализации сервлета создайте фиктивные запросы и ответы и введите их в метод service() servt. Если запрос направлен правильно, вы можете проверить результаты, как написано в ложном ответе.

1 голос
/ 14 октября 2012

В предстоящей весне 3.2 (доступен SNAPSHOT) или с помощью spring-test-mvc (https://github.com/SpringSource/spring-test-mvc) вы можете сделать это следующим образом:

сначала мы эмулируем валидацию, так как не хотим тестировать валидатор, просто хотим знать, вызывается ли валидация.

public class LocalValidatorFactoryBeanMock extends LocalValidatorFactoryBean
{
    private boolean fakeErrors;

    public void fakeErrors ( )
    {
        this.fakeErrors = true;
    }

    @Override
    public boolean supports ( Class<?> clazz )
    {
        return true;
    }

    @Override
    public void validate ( Object target, Errors errors, Object... validationHints )
    {
        if (fakeErrors)
        {
            errors.reject("error");
        }
    }
}

это наш тестовый класс:

@RunWith(SpringJUnit4ClassRunner.class)
@WebAppConfiguration
@ContextConfiguration
public class RegisterControllerTest
{
 @Autowired
 private WebApplicationContext  wac;
 private MockMvc mockMvc;

     @Autowired
     @InjectMocks
     private RegisterController registerController;

     @Autowired
     private LocalValidatorFactoryBeanMock  validator;

  @Before
  public void setup ( )
  {
     this.mockMvc = MockMvcBuilders.webAppContextSetup(this.wac).build();
     // if you want to inject mocks into your controller
             MockitoAnnotations.initMocks(this);
  }

    @Test
    public void testPostValidationError ( ) throws Exception
    {
        validator.fakeErrors();
        MockHttpServletRequestBuilder post = post("/info/register");
        post.param("name", "Bob");
        ResultActions result = getMockMvc().perform(post);
            // no redirect as we have errors
        result.andExpect(view().name("info/register"));
    }

    @Configuration
    @Import(DispatcherServletConfig.class)
    static class Config extends WebMvcConfigurerAdapter
    {
        @Override
        public Validator getValidator ( )
        {
            return new LocalValidatorFactoryBeanMock();
        }

        @Bean
        RegisterController registerController ( )
        {
            return new RegisterController();
        }
    }
}
...