Напишите JUnit-тест для @ExceptionHandler

Я пишу службу Rest с использованием Spring MVC. Вот контур класса:

 @Controller
 public class MyController{

     @RequestMapping(..)
     public void myMethod(...) throws NotAuthorizedException{...}

     @ExceptionHandler(NotAuthorizedException.class)
     @ResponseStatus(value=HttpStatus.UNAUTHORIZED, reason="blah")
     public void handler(...){...}
 }

Я написал свои модульные тесты, используя дизайн здесь. Тест в основном выглядит следующим образом:

@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(....)
public class mytest{

    MockHttpServletRequest requestMock;
    MockHttpServletResponse responseMock;
    AnnotationMethodHandlerAdapter handlerAdapter;

@Before
public void setUp() {
    requestMock = new MockHttpServletRequest();
    requestMock.setContentType(MediaType.APPLICATION_JSON_VALUE);
    requestMock.addHeader(HttpHeaders.ACCEPT, MediaType.APPLICATION_JSON_VALUE);

    responseMock = new MockHttpServletResponse();

    handlerAdapter = new AnnotationMethodHandlerAdapter();
}

@Test
public void testExceptionHandler(){
    // setup ....
    handlerAdapter.handle(...);

    // verify
    // I would like to do the following
    assertThat(responseMock.getStatus(), is(HttpStatus.UNAUTHORIZED.value()));
}

}

Однако вызов handle вызывает NotAuthorizedException. Я прочитал, что по дизайну, чтобы иметь возможность unit test, чтобы метод выбрал соответствующее исключение, однако я хотел бы написать автоматизированный тест, чтобы структура обрабатывала это исключение соответствующим образом и что тестируемый класс реализовал обработчик соответственно. Есть ли способ сделать это?

Помните, что у меня нет доступа к фактическому коду в месте, где я мог бы его опубликовать.

Кроме того, я ограничен (по несчастью) до Spring 3.0.5 или 3.1.2.

Ответ 1

Рассмотрим использование Spring 3.2 и его mvc-test-framework

import static org.springframework.test.web.servlet.setup.MockMvcBuilders.*;
import static org.springframework.test.web.servlet.request.MockMvcRequestBuilders.*;
import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.*;

@RunWith(SpringJUnit4ClassRunner.class)
@WebAppConfiguration
@ContextConfiguration("file:src/main/webapp/WEB-INF/spring/appServlet/servlet-context.xml")
public class WebMvcTest {

    @Autowired
    private WebApplicationContext wac;

    private MockMvc mockMvc;

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

    @Test
    public void getFoo() throws Exception {
        this.mockMvc.perform(
            get("/testx")
            .contentType(MediaType.APPLICATION_JSON)
            .accept(MediaType.APPLICATION_JSON)
            )
            .andExpect(status().isUnauthorized());
    }
}

Код контроллера

@Controller
public class MyController {

    public class MyException extends RuntimeException {
    };

    @RequestMapping("/testx")
    public void myMethod() {
        throw new MyException();

    }

    @ExceptionHandler(MyException.class)
    @ResponseStatus(value = HttpStatus.UNAUTHORIZED, reason = "blah")
    public void handler() {
        System.out.println("handler processed");
    }
}

Этот "тест" проходит хорошо.

Отказ от ответственности: в настоящее время я являюсь noob в Spring тестировании MVC, на самом деле это мой первый тест.
upd: Спасибо The Drake за исправление. Суб >

Ответ 2

Аннотировать свой контроллер обработки исключений @ControllerAdvice вместо @Controller.

Как отметил Борис Треухов при добавлении аннотации @ExceptionHandler к методу в контроллере, который генерирует исключение, он будет работать, но только с этого конкретного контроллера.

@ControllerAdvice позволит использовать методы обработки исключений для всего приложения не только для одного конкретного контроллера.

Ответ 3

Вы можете изменить @Test на

@Test(expected=NotAuthorizedException.class)

Это вернет true, если внутренности будут вызывать это исключение и false в противном случае.

Это также сделает assertThat() ненужным. Вы можете написать второй тест, который ловит NotAuthorizedException, тогда вы можете проверить responseMock в этом состоянии.