Модульный тест Spring MissingServletRequestParameterException Ответ JSON

У меня есть метод POST в контроллере Spring Boot Rest следующим образом

@RequestMapping(value="/post/action/bookmark", method=RequestMethod.POST)
public @ResponseBody Map<String, String> bookmarkPost(
        @RequestParam(value="actionType",required=true) String actionType,
        @RequestParam(value="postId",required=true) String postId,
        @CurrentUser User user) throws Exception{
    return service.bookmarkPost(postId, actionType, user);
}

теперь, если я проверю с отсутствующим параметром в Postman, я получу ответ 400 http и тело JSON:

{
  "timestamp": "2015-07-20",
  "status": 400,
  "error": "Bad Request",
  "exception": "org.springframework.web.bind.MissingServletRequestParameterException",
  "message": "Required String parameter 'actionType' is not present",
  "path": "/post/action/bookmark"
}

до сих пор все в порядке, но когда я пытаюсь выполнить модульное тестирование, я не получаю ответ JSON

@Test
public void bookmarkMissingActionTypeParam() throws Exception{
    // @formatter:off
    mockMvc.perform(
                post("/post/action/bookmark")
                    .accept(MediaType.APPLICATION_JSON)
                    .param("postId", "55ab8831036437e96e8250b6")
                    )
            .andExpect(status().isBadRequest())
            .andExpect(jsonPath("$.exception", containsString("MissingServletRequestParameterException")));
    // @formatter:on
}

тест не проходит и выдает

java.lang.IllegalArgumentException: json can not be null or empty

Я сделал .andDo(print()) и обнаружил, что в ответе нет тела

MockHttpServletResponse:
          Status = 400
   Error message = Required String parameter 'actionType' is not present
         Headers = {X-Content-Type-Options=[nosniff], X-XSS-Protection=[1; mode=block], Cache-Control=[no-cache, no-store], Pragma=[no-cache], Expires=[1], X-Frame-Options=[DENY]}
    Content type = null
            Body = 
   Forwarded URL = null
  Redirected URL = null
         Cookies = []

почему я не получаю ответ JSON при модульном тестировании моего контроллера, но получаю его при ручном тестировании с использованием Postman или cUrl?

РЕДАКТИРОВАТЬ: я добавил @WebIntegrationTest, но получил ту же ошибку:

import static org.hamcrest.Matchers.containsString;
import static org.springframework.test.web.servlet.request.MockMvcRequestBuilders.post;
import static org.springframework.test.web.servlet.result.MockMvcResultHandlers.print;
import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.jsonPath;
import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.status;
import org.junit.Before;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.SpringApplicationConfiguration;
import org.springframework.boot.test.WebIntegrationTest;
import org.springframework.http.MediaType;
import org.springframework.security.web.FilterChainProxy;
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
import org.springframework.test.web.servlet.MockMvc;
import org.springframework.test.web.servlet.setup.MockMvcBuilders;
import org.springframework.web.context.WebApplicationContext;

@RunWith(SpringJUnit4ClassRunner.class)
@SpringApplicationConfiguration(classes = RestApplication.class)
@WebIntegrationTest
public class PostControllerTest {

    private MockMvc mockMvc;

    @Autowired
    private WebApplicationContext webApplicationContext;

    @Autowired
    private FilterChainProxy springSecurityFilterChain;

    @Before
    public void setUp() {
        mockMvc = MockMvcBuilders.webAppContextSetup(webApplicationContext)
                .addFilter(springSecurityFilterChain)
                .build();
    }  

    @Test
    public void bookmarkMissingActionTypeParam() throws Exception{
        // @formatter:off
        mockMvc.perform(
                    post("/post/action/bookmark")
                        .accept(MediaType.APPLICATION_JSON)
                        .param("postId", "55ab8831036437e96e8250b6")
                        )
                .andDo(print())
                .andExpect(status().isBadRequest())
                .andExpect(jsonPath("$.exception", containsString("MissingServletRequestParameterException")));
        // @formatter:on
    }
}

person Youssef Al Muhaidib    schedule 22.07.2015    source источник
comment
У меня точно такая же проблема с ответом без ошибок: у вас есть какие-либо успехи в этом?   -  person Roberto Lo Giacco    schedule 16.11.2015
comment
Нет, еще нет... Я пропустил тестирование этих кейсов!   -  person Youssef Al Muhaidib    schedule 16.11.2015


Ответы (2)


Это связано с тем, что Spring Boot автоматически настроил обработчик исключений org.springframework.boot.autoconfigure.web.BasicErrorController, которого, вероятно, нет в ваших модульных тестах. Способ получить его — использовать аннотации, связанные с поддержкой тестирования Spring Boot:

@SpringApplicationConfiguration
@WebIntegrationTest

Дополнительные сведения см. в здесь

Обновление: вы абсолютно правы, поведение в пользовательском интерфейсе и в тесте сильно отличается, страницы ошибок, которые реагируют на коды состояния, неправильно подключены в тестовой среде без сервлета. Улучшение этого поведения может быть хорошей ошибкой для Spring MVC и/или Spring Boot.

На данный момент у меня есть обходной путь, который имитирует поведение BasicErrorController следующим образом:

@RunWith(SpringJUnit4ClassRunner.class)
@SpringApplicationConfiguration(classes = {RestApplication.class, TestConfiguration.class})
@WebIntegrationTest
public class PostControllerTest {

    private MockMvc mockMvc;

    @Autowired
    private WebApplicationContext webApplicationContext;

    @Autowired
    private FilterChainProxy springSecurityFilterChain;

    @Before
    public void setUp() {
        mockMvc = MockMvcBuilders.webAppContextSetup(webApplicationContext)
                .addFilter(springSecurityFilterChain)
                .build();
    }  

    @Test
    public void bookmarkMissingActionTypeParam() throws Exception{
        // @formatter:off
        mockMvc.perform(
                    post("/post/action/bookmark")
                        .accept(MediaType.APPLICATION_JSON)
                        .param("postId", "55ab8831036437e96e8250b6")
                        )
                .andDo(print())
                .andExpect(status().isBadRequest())
                .andExpect(jsonPath("$.exception", containsString("MissingServletRequestParameterException")));
        // @formatter:on
    }
}
     @Configuration
    public static class TestConfiguration {


        @Bean
        public ErrorController errorController(ErrorAttributes errorAttributes) {
            return new ErrorController(errorAttributes);
        }
    }
@ControllerAdvice
class ErrorController extends BasicErrorController {

    public ErrorController(ErrorAttributes errorAttributes) {
        super(errorAttributes);
    }

    @Override
    @ExceptionHandler(Exception.class)
    @ResponseBody
    public ResponseEntity<Map<String, Object>> error(HttpServletRequest request) {
        return super.error(request);
    }
}

Что я здесь делаю, так это добавляю ControllerAdvice, который обрабатывает поток исключений и делегирует обратно в BasicErrorController. Это, по крайней мере, сделало бы поведение последовательным для вас.

person Biju Kunjummen    schedule 22.07.2015
comment
Я добавил @WebIntegrationTest, но получил ту же ошибку, я добавил полный тестовый класс выше - person Youssef Al Muhaidib; 23.07.2015
comment
Просто для информации, у вашего RestApplication есть аннотация @EnableAutoConfiguration. - person Biju Kunjummen; 23.07.2015
comment
нет, используя только «@SpringBootApplication»… разве это не эквивалентно? - person Youssef Al Muhaidib; 23.07.2015
comment
Вы правы, я добавил обновление с потенциальным обходным путем. - person Biju Kunjummen; 24.07.2015
comment
попытался поместить TestConfiguration вне PostControllerTest (но в том же файле и не скомпилировался из-за статики, то же самое, если бы я добавил TestConfiguration внутри PostControllerTest (@SpringApplicationConfiguration не может преобразовать TestConfiguration в тип), в последний раз я добавил TestConfiguration в отдельный файл и удалил статику, я получаю Вызвано: java.lang.NoSuchMethodException: com.rest.ErrorController.‹init›() - person Youssef Al Muhaidib; 29.07.2015
comment
Я столкнулся с той же проблемой и до сих пор не исправлен - person Muhammad Hewedy; 15.09.2016

Первоначально он должен исправить ошибку с помощью тега @ResponseBody при определении метода вашего контроллера REST. это исправит ошибку json в тестовом классе. Но, поскольку вы используете весеннюю загрузку, вы определите класс контроллера с @RestController, и он должен автоматически позаботиться об ошибке без определения тегов @Controller и @ResponseType.

person Vishal    schedule 13.04.2016
comment
Я столкнулся с той же проблемой, и этот ответ не помогает, @Biju правильно понимает их проблему. - person Muhammad Hewedy; 15.09.2016