responseentity objects mkyong headers getforobject example create java spring rest resttemplate

java - objects - spring create resttemplate



REST cliente restTemplate no puede obtener Colección de objetos (4)

Yo uso Spring restTemplate. Hice un servicio REST y un cliente como prueba unitaria en una aplicación separada. Tengo un método que devuelve la Lista de usuarios y el método para la creación del usuario:

@GET @Produces({ MediaType.APPLICATION_XML, MediaType.APPLICATION_JSON, MediaType.TEXT_XML }) @Path("/all") public Response getAllUsers() { List<User> list = dao.getAll(); GenericEntity<List<User>> result = new GenericEntity<List<User>>(list) { }; return Response.status(Status.OK).entity(result).build(); }

Si solicito mostrarme todos los usuarios en el navegador, me muestra un xml. Está bien. Pero, cuando trato de usar esto:

@Test public void testGetAll() { List list = new RestTemplate().getForObject(URL + "all", List.class); System.out.println(list); }

tengo

WARNING: GET request for "http://localhost:8080/REST/all" resulted in 500 (Internal Server Error); invoking error handler

Intenté depurar esto. Sin excepciones durante el método funciona. Y el navegador me muestra el xml con los usuarios. ¿Qué puede estar mal?

Además, deseo saber cómo puedo obtener el código de estado o el mensaje del objeto de la plantilla (para la prueba).

Gracias por tus respuestas.

EDITADO:

Modifiqué mi método de prueba:

@Test public void testGetAll() { RestTemplate template = new RestTemplate(); List<HttpMessageConverter<?>> messageConverters = new ArrayList<HttpMessageConverter<?>>(); Jaxb2RootElementHttpMessageConverter jaxbMessageConverter = new Jaxb2RootElementHttpMessageConverter(); List<MediaType> mediaTypes = new ArrayList<MediaType>(); mediaTypes.add(MediaType.APPLICATION_XML); jaxbMessageConverter.setSupportedMediaTypes(mediaTypes); messageConverters.add(jaxbMessageConverter); template.setMessageConverters(messageConverters); List list = template.getForObject(URL + "all", ArrayList.class); System.out.println(list); }

Y tengo una excepción:

org.springframework.web.client.RestClientException: Could not extract response: no suitable HttpMessageConverter found for response type [class java.util.ArrayList] and content type [application/xml] at org.springframework.web.client.HttpMessageConverterExtractor.extractData(HttpMessageConverterExtractor.java:107) at org.springframework.web.client.RestTemplate.doExecute(RestTemplate.java:496) at org.springframework.web.client.RestTemplate.execute(RestTemplate.java:452) at org.springframework.web.client.RestTemplate.getForObject(RestTemplate.java:222) at com.nixsolutions.web.service.rest.UserRESTServiceTest.testGetAll(UserRESTServiceTest.java:61) at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method) at sun.reflect.NativeMethodAccessorImpl.invoke(Unknown Source) at sun.reflect.DelegatingMethodAccessorImpl.invoke(Unknown Source) at java.lang.reflect.Method.invoke(Unknown Source) at org.junit.runners.model.FrameworkMethod$1.runReflectiveCall(FrameworkMethod.java:47) at org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:12) at org.junit.runners.model.FrameworkMethod.invokeExplosively(FrameworkMethod.java:44) at org.junit.internal.runners.statements.InvokeMethod.evaluate(InvokeMethod.java:17) at org.junit.runners.ParentRunner.runLeaf(ParentRunner.java:271) at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:70) at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:50) at org.junit.runners.ParentRunner$3.run(ParentRunner.java:238) at org.junit.runners.ParentRunner$1.schedule(ParentRunner.java:63) at org.junit.runners.ParentRunner.runChildren(ParentRunner.java:236) at org.junit.runners.ParentRunner.access$000(ParentRunner.java:53) at org.junit.runners.ParentRunner$2.evaluate(ParentRunner.java:229) at org.junit.runners.ParentRunner.run(ParentRunner.java:309) at org.eclipse.jdt.internal.junit4.runner.JUnit4TestReference.run(JUnit4TestReference.java:50) at org.eclipse.jdt.internal.junit.runner.TestExecution.run(TestExecution.java:38) at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:467) at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:683) at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.run(RemoteTestRunner.java:390) at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.main(RemoteTestRunner.java:197)


Cuando configura el Jaxb2RootElementHttpMessageConverter , anula los convertidores predeterminados que vienen con RestTemplate . Uno de los conversores predeterminados (creo que es el convertidor de cadenas) puede manejar el tipo de text/xml . Elimine todo el Jaxb2RootElementHttpMessageConverter pero deje esa parte cuando espera ArrayList.class y no List.class y esto funcionará:

@Test public void testGetAll() { RestTemplate template = new RestTemplate(); List list = template.getForObject(URL + "all", ArrayList.class); System.out.println(list); }

Es posible que también deba agregar un encabezado de accept para elegir el uso de text/xml y no uno de los otros tipos producidos:

HttpHeaders requestHeaders = new HttpHeaders(); requestHeaders.setAccept(Arrays.asList(new MediaType[] {MediaType.TEXT_XML}));

Y use exchange con RestTemplate (en lugar de getForObject ):

List list = template.exchange(URL + "all", new HttpEntity<String>(requestHeaders()), ArrayList.class);


Necesita utilizar una implementación concreta de List , por ejemplo, puede usar ArrayList , consulte este ejemplo:

ResponseEntity<? extends ArrayList<User>> responseEntity = restTemplate.getForEntity(restEndPointUrl, (Class<? extends ArrayList<User>>)ArrayList.class, userId);

incluso funciona para una configuración completamente genérica:

ResponseEntity<? extends ArrayList<HashMap<String,Object>>> responseEntity = restTemplate.getForEntity(restEndPointUrl, (Class<? extends ArrayList<HashMap<String,Object>>>)ArrayList.class, parameterId);


Podría usar restTemplate.getForEntity() . Le devolverá ResponseEntity con toda la información de respuesta (incluido el estado).


tal vez de esta manera ...

RestTemplate template = new RestTemplate(true); ResponseEntity<TblGps[]> responseEntity = restTemplate.getForEntity(urlGETList, TblGps[].class); TblGps[]=responseEntity.getBody();