Concluído 406 NOT_ACCEPTABLE - Testando WebLayer no SpringBoot

Dec 07 2020

Eu tenho esse método em um aplicativo Spring Boot v2.1.0.RELEASE.

@GetMapping(value = "/wildProject", produces = MediaType.APPLICATION_JSON_VALUE)
public ResponseEntity<List<WildProject>> getList(HttpServletRequest request,
        HttpServletResponse response)
        throws Exception {

    List<WildProject> list = authorisationService.getList();

    System.out.println("-----------------");
    System.out.println(list);
    System.out.println("-----------------");

    return ok().body(list);


} 

e este teste:

 this.mockMvc.perform(get("/wildProject")
  //.accept(MediaType.APPLICATION_JSON_UTF8_VALUE))
  // .andDo(print())
  .andExpect(content().contentType(MediaType.APPLICATION_JSON_VALUE))
  .andExpect(status().isOk());

e este é o resultado do teste:

20:03:38.253 [main] DEBUG o.s.w.s.m.m.a.HttpEntityMethodProcessor - Using 'application/json', given [*/*] and supported [application/json]
20:03:38.255 [main] WARN  o.s.w.s.m.s.DefaultHandlerExceptionResolver - Resolved [org.springframework.web.HttpMediaTypeNotAcceptableException: Could not find acceptable representation]
20:03:38.256 [main] DEBUG o.s.t.w.s.TestDispatcherServlet - Completed 406 NOT_ACCEPTABLE

MockHttpServletRequest:
      HTTP Method = GET
      Request URI = /wildProject
       Parameters = {}
          Headers = {Content-Type=[application/json;charset=UTF-8]}
             Body = null
    Session Attrs = {}

Handler:
             Type = com.bonansa.controller.AuthorisationController
           Method = public org.springframework.http.ResponseEntity<java.util.List<com.bonansa.WildProject>> com.bonansa.controller.AuthorisationController.getList() throws java.lang.Exception

Async:
    Async started = false
     Async result = null

Resolved Exception:
             Type = org.springframework.web.HttpMediaTypeNotAcceptableException

ModelAndView:
        View name = null
             View = null
            Model = null

FlashMap:
       Attributes = null

MockHttpServletResponse:
           Status = 406
    Error message = null
          Headers = {}
     Content type = null
             Body = 
    Forwarded URL = null
   Redirected URL = null
          Cookies = []

e

@JsonSerialize(as = IWildProject.class)
public interface IWildProject {
..
}

Respostas

s7vr Dec 19 2020 at 04:59

Seu problema parece duplo.

O problema subjacente é que você não registrou o conversor de mensagem certo para escrever uma resposta json. Isso está oculto por causa do bug onde o servidor deveria retornar 500 em vez de 406 para configuração incorreta, pois o controlador especificou a anotação de produção e o cliente aceitou o cabeçalho de todos.

Mais detalhes - https://github.com/spring-projects/spring-framework/issues/23287

Isso não é mais um problema na versão mais recente, apenas com a versão 2.1.x de boot de primavera

2 x80486 Dec 08 2020 at 00:48

Você não precisa definir o Content-Typecabeçalho para GETsolicitações, pois você não está enviando nada, mas pedindo algo. O Acceptcabeçalho é o que estamos procurando neste caso.

Além disso, MediaType.APPLICATION_JSON_VALUEnão corresponderá MediaType.APPLICATION_JSON_UTF8_VALUE.

Eu recomendaria refatorar esse trecho de código em:

this.mockMvc.perform(get("/wildProject")
  .accept(MediaType.APPLICATION_JSON_VALUE))
  // .andDo(print())
  .andExpect(content().contentType(MediaType.APPLICATION_JSON_VALUE))
  .andExpect(status().isOk());