Forbid null converters in RestTemplate & HttpMessageConverterExtractor
Prior to this commit, RestTemplate and HttpMessageConverterExtractor did not validate that the supplied HttpMessageConverter list contained no null elements, which can lead to a NullPointerException when the converters are accessed. This commit improves the user experience by failing immediately if the supplied HttpMessageConverter list contains a null element. This applies to constructors for RestTemplate and HttpMessageConverterExtractor as well as to RestTemplate#setMessageConverters(). Note, however, that RestTemplate#getMessageConverters() returns a mutable list. Thus, if a user modifies that list so that it contains null values, that will still lead to a NullPointerException when the converters are accessed. This commit also introduces noNullElements() variants for collections in org.springframework.util.Assert. Closes gh-23151
This commit is contained in:
@@ -19,7 +19,7 @@ package org.springframework.web.client;
|
||||
import java.io.ByteArrayInputStream;
|
||||
import java.io.IOException;
|
||||
import java.lang.reflect.Type;
|
||||
import java.util.ArrayList;
|
||||
import java.util.Arrays;
|
||||
import java.util.List;
|
||||
|
||||
import org.junit.Test;
|
||||
@@ -34,8 +34,10 @@ import org.springframework.http.converter.GenericHttpMessageConverter;
|
||||
import org.springframework.http.converter.HttpMessageConverter;
|
||||
import org.springframework.http.converter.HttpMessageNotReadableException;
|
||||
|
||||
import static java.util.Arrays.asList;
|
||||
import static org.assertj.core.api.Assertions.assertThat;
|
||||
import static org.assertj.core.api.Assertions.assertThatExceptionOfType;
|
||||
import static org.assertj.core.api.Assertions.assertThatIllegalArgumentException;
|
||||
import static org.mockito.ArgumentMatchers.any;
|
||||
import static org.mockito.ArgumentMatchers.eq;
|
||||
import static org.mockito.BDDMockito.given;
|
||||
@@ -46,18 +48,30 @@ import static org.mockito.Mockito.mock;
|
||||
*
|
||||
* @author Arjen Poutsma
|
||||
* @author Brian Clozel
|
||||
* @author Sam Brannen
|
||||
*/
|
||||
public class HttpMessageConverterExtractorTests {
|
||||
|
||||
private HttpMessageConverterExtractor<?> extractor;
|
||||
|
||||
@SuppressWarnings("unchecked")
|
||||
private final HttpMessageConverter<String> converter = mock(HttpMessageConverter.class);
|
||||
private final HttpMessageConverterExtractor<?> extractor = new HttpMessageConverterExtractor<>(String.class, asList(converter));
|
||||
private final MediaType contentType = MediaType.TEXT_PLAIN;
|
||||
private final HttpHeaders responseHeaders = new HttpHeaders();
|
||||
private final ClientHttpResponse response = mock(ClientHttpResponse.class);
|
||||
|
||||
|
||||
@Test
|
||||
public void constructorPreconditions() {
|
||||
assertThatIllegalArgumentException()
|
||||
.isThrownBy(() -> new HttpMessageConverterExtractor<>(String.class, (List<HttpMessageConverter<?>>) null))
|
||||
.withMessage("'messageConverters' must not be empty");
|
||||
assertThatIllegalArgumentException()
|
||||
.isThrownBy(() -> new HttpMessageConverterExtractor<>(String.class, Arrays.asList(null, this.converter)))
|
||||
.withMessage("'messageConverters' must not contain null elements");
|
||||
}
|
||||
|
||||
@Test
|
||||
public void noContent() throws IOException {
|
||||
HttpMessageConverter<?> converter = mock(HttpMessageConverter.class);
|
||||
extractor = new HttpMessageConverterExtractor<>(String.class, createConverterList(converter));
|
||||
given(response.getRawStatusCode()).willReturn(HttpStatus.NO_CONTENT.value());
|
||||
|
||||
Object result = extractor.extractData(response);
|
||||
@@ -66,8 +80,6 @@ public class HttpMessageConverterExtractorTests {
|
||||
|
||||
@Test
|
||||
public void notModified() throws IOException {
|
||||
HttpMessageConverter<?> converter = mock(HttpMessageConverter.class);
|
||||
extractor = new HttpMessageConverterExtractor<>(String.class, createConverterList(converter));
|
||||
given(response.getRawStatusCode()).willReturn(HttpStatus.NOT_MODIFIED.value());
|
||||
|
||||
Object result = extractor.extractData(response);
|
||||
@@ -76,8 +88,6 @@ public class HttpMessageConverterExtractorTests {
|
||||
|
||||
@Test
|
||||
public void informational() throws IOException {
|
||||
HttpMessageConverter<?> converter = mock(HttpMessageConverter.class);
|
||||
extractor = new HttpMessageConverterExtractor<>(String.class, createConverterList(converter));
|
||||
given(response.getRawStatusCode()).willReturn(HttpStatus.CONTINUE.value());
|
||||
|
||||
Object result = extractor.extractData(response);
|
||||
@@ -86,10 +96,7 @@ public class HttpMessageConverterExtractorTests {
|
||||
|
||||
@Test
|
||||
public void zeroContentLength() throws IOException {
|
||||
HttpMessageConverter<?> converter = mock(HttpMessageConverter.class);
|
||||
HttpHeaders responseHeaders = new HttpHeaders();
|
||||
responseHeaders.setContentLength(0);
|
||||
extractor = new HttpMessageConverterExtractor<>(String.class, createConverterList(converter));
|
||||
given(response.getRawStatusCode()).willReturn(HttpStatus.OK.value());
|
||||
given(response.getHeaders()).willReturn(responseHeaders);
|
||||
|
||||
@@ -98,11 +105,7 @@ public class HttpMessageConverterExtractorTests {
|
||||
}
|
||||
|
||||
@Test
|
||||
@SuppressWarnings("unchecked")
|
||||
public void emptyMessageBody() throws IOException {
|
||||
HttpMessageConverter<String> converter = mock(HttpMessageConverter.class);
|
||||
HttpHeaders responseHeaders = new HttpHeaders();
|
||||
extractor = new HttpMessageConverterExtractor<>(String.class, createConverterList(converter));
|
||||
given(response.getRawStatusCode()).willReturn(HttpStatus.OK.value());
|
||||
given(response.getHeaders()).willReturn(responseHeaders);
|
||||
given(response.getBody()).willReturn(new ByteArrayInputStream("".getBytes()));
|
||||
@@ -112,11 +115,7 @@ public class HttpMessageConverterExtractorTests {
|
||||
}
|
||||
|
||||
@Test // gh-22265
|
||||
@SuppressWarnings("unchecked")
|
||||
public void nullMessageBody() throws IOException {
|
||||
HttpMessageConverter<String> converter = mock(HttpMessageConverter.class);
|
||||
HttpHeaders responseHeaders = new HttpHeaders();
|
||||
extractor = new HttpMessageConverterExtractor<>(String.class, createConverterList(converter));
|
||||
given(response.getRawStatusCode()).willReturn(HttpStatus.OK.value());
|
||||
given(response.getHeaders()).willReturn(responseHeaders);
|
||||
given(response.getBody()).willReturn(null);
|
||||
@@ -126,14 +125,9 @@ public class HttpMessageConverterExtractorTests {
|
||||
}
|
||||
|
||||
@Test
|
||||
@SuppressWarnings("unchecked")
|
||||
public void normal() throws IOException {
|
||||
HttpMessageConverter<String> converter = mock(HttpMessageConverter.class);
|
||||
HttpHeaders responseHeaders = new HttpHeaders();
|
||||
MediaType contentType = MediaType.TEXT_PLAIN;
|
||||
responseHeaders.setContentType(contentType);
|
||||
String expected = "Foo";
|
||||
extractor = new HttpMessageConverterExtractor<>(String.class, createConverterList(converter));
|
||||
given(response.getRawStatusCode()).willReturn(HttpStatus.OK.value());
|
||||
given(response.getHeaders()).willReturn(responseHeaders);
|
||||
given(response.getBody()).willReturn(new ByteArrayInputStream(expected.getBytes()));
|
||||
@@ -145,32 +139,26 @@ public class HttpMessageConverterExtractorTests {
|
||||
}
|
||||
|
||||
@Test
|
||||
@SuppressWarnings("unchecked")
|
||||
public void cannotRead() throws IOException {
|
||||
HttpMessageConverter<String> converter = mock(HttpMessageConverter.class);
|
||||
HttpHeaders responseHeaders = new HttpHeaders();
|
||||
MediaType contentType = MediaType.TEXT_PLAIN;
|
||||
responseHeaders.setContentType(contentType);
|
||||
extractor = new HttpMessageConverterExtractor<>(String.class, createConverterList(converter));
|
||||
given(response.getRawStatusCode()).willReturn(HttpStatus.OK.value());
|
||||
given(response.getHeaders()).willReturn(responseHeaders);
|
||||
given(response.getBody()).willReturn(new ByteArrayInputStream("Foobar".getBytes()));
|
||||
given(converter.canRead(String.class, contentType)).willReturn(false);
|
||||
assertThatExceptionOfType(RestClientException.class).isThrownBy(() ->
|
||||
extractor.extractData(response));
|
||||
assertThatExceptionOfType(RestClientException.class).isThrownBy(() -> extractor.extractData(response));
|
||||
}
|
||||
|
||||
@Test
|
||||
@SuppressWarnings("unchecked")
|
||||
public void generics() throws IOException {
|
||||
GenericHttpMessageConverter<String> converter = mock(GenericHttpMessageConverter.class);
|
||||
HttpHeaders responseHeaders = new HttpHeaders();
|
||||
MediaType contentType = MediaType.TEXT_PLAIN;
|
||||
responseHeaders.setContentType(contentType);
|
||||
String expected = "Foo";
|
||||
ParameterizedTypeReference<List<String>> reference = new ParameterizedTypeReference<List<String>>() {};
|
||||
Type type = reference.getType();
|
||||
extractor = new HttpMessageConverterExtractor<List<String>>(type, createConverterList(converter));
|
||||
|
||||
GenericHttpMessageConverter<String> converter = mock(GenericHttpMessageConverter.class);
|
||||
HttpMessageConverterExtractor<?> extractor = new HttpMessageConverterExtractor<List<String>>(type, asList(converter));
|
||||
|
||||
given(response.getRawStatusCode()).willReturn(HttpStatus.OK.value());
|
||||
given(response.getHeaders()).willReturn(responseHeaders);
|
||||
given(response.getBody()).willReturn(new ByteArrayInputStream(expected.getBytes()));
|
||||
@@ -182,48 +170,28 @@ public class HttpMessageConverterExtractorTests {
|
||||
}
|
||||
|
||||
@Test // SPR-13592
|
||||
@SuppressWarnings("unchecked")
|
||||
public void converterThrowsIOException() throws IOException {
|
||||
HttpMessageConverter<String> converter = mock(HttpMessageConverter.class);
|
||||
HttpHeaders responseHeaders = new HttpHeaders();
|
||||
MediaType contentType = MediaType.TEXT_PLAIN;
|
||||
responseHeaders.setContentType(contentType);
|
||||
extractor = new HttpMessageConverterExtractor<>(String.class, createConverterList(converter));
|
||||
given(response.getRawStatusCode()).willReturn(HttpStatus.OK.value());
|
||||
given(response.getHeaders()).willReturn(responseHeaders);
|
||||
given(response.getBody()).willReturn(new ByteArrayInputStream("Foobar".getBytes()));
|
||||
given(converter.canRead(String.class, contentType)).willReturn(true);
|
||||
given(converter.read(eq(String.class), any(HttpInputMessage.class))).willThrow(IOException.class);
|
||||
assertThatExceptionOfType(RestClientException.class).isThrownBy(() ->
|
||||
extractor.extractData(response))
|
||||
assertThatExceptionOfType(RestClientException.class).isThrownBy(() -> extractor.extractData(response))
|
||||
.withMessageContaining("Error while extracting response for type [class java.lang.String] and content type [text/plain]")
|
||||
.withCauseInstanceOf(IOException.class);
|
||||
}
|
||||
|
||||
@Test // SPR-13592
|
||||
@SuppressWarnings("unchecked")
|
||||
public void converterThrowsHttpMessageNotReadableException() throws IOException {
|
||||
HttpMessageConverter<String> converter = mock(HttpMessageConverter.class);
|
||||
HttpHeaders responseHeaders = new HttpHeaders();
|
||||
MediaType contentType = MediaType.TEXT_PLAIN;
|
||||
responseHeaders.setContentType(contentType);
|
||||
extractor = new HttpMessageConverterExtractor<>(String.class, createConverterList(converter));
|
||||
given(response.getRawStatusCode()).willReturn(HttpStatus.OK.value());
|
||||
given(response.getHeaders()).willReturn(responseHeaders);
|
||||
given(response.getBody()).willReturn(new ByteArrayInputStream("Foobar".getBytes()));
|
||||
given(converter.canRead(String.class, contentType)).willThrow(HttpMessageNotReadableException.class);
|
||||
assertThatExceptionOfType(RestClientException.class).isThrownBy(() ->
|
||||
extractor.extractData(response))
|
||||
assertThatExceptionOfType(RestClientException.class).isThrownBy(() -> extractor.extractData(response))
|
||||
.withMessageContaining("Error while extracting response for type [class java.lang.String] and content type [text/plain]")
|
||||
.withCauseInstanceOf(HttpMessageNotReadableException.class);
|
||||
|
||||
}
|
||||
|
||||
private List<HttpMessageConverter<?>> createConverterList(HttpMessageConverter<?> converter) {
|
||||
List<HttpMessageConverter<?>> converters = new ArrayList<>(1);
|
||||
converters.add(converter);
|
||||
return converters;
|
||||
}
|
||||
|
||||
|
||||
}
|
||||
|
||||
@@ -49,6 +49,7 @@ import org.springframework.web.util.DefaultUriBuilderFactory;
|
||||
|
||||
import static org.assertj.core.api.Assertions.assertThat;
|
||||
import static org.assertj.core.api.Assertions.assertThatExceptionOfType;
|
||||
import static org.assertj.core.api.Assertions.assertThatIllegalArgumentException;
|
||||
import static org.mockito.ArgumentMatchers.any;
|
||||
import static org.mockito.ArgumentMatchers.eq;
|
||||
import static org.mockito.BDDMockito.given;
|
||||
@@ -65,9 +66,12 @@ import static org.springframework.http.HttpMethod.PUT;
|
||||
import static org.springframework.http.MediaType.parseMediaType;
|
||||
|
||||
/**
|
||||
* Unit tests for {@link RestTemplate}.
|
||||
*
|
||||
* @author Arjen Poutsma
|
||||
* @author Rossen Stoyanchev
|
||||
* @author Brian Clozel
|
||||
* @author Sam Brannen
|
||||
*/
|
||||
@SuppressWarnings("unchecked")
|
||||
public class RestTemplateTests {
|
||||
@@ -98,6 +102,25 @@ public class RestTemplateTests {
|
||||
template.setErrorHandler(errorHandler);
|
||||
}
|
||||
|
||||
@Test
|
||||
public void constructorPreconditions() {
|
||||
assertThatIllegalArgumentException()
|
||||
.isThrownBy(() -> new RestTemplate((List<HttpMessageConverter<?>>) null))
|
||||
.withMessage("At least one HttpMessageConverter is required");
|
||||
assertThatIllegalArgumentException()
|
||||
.isThrownBy(() -> new RestTemplate(Arrays.asList(null, this.converter)))
|
||||
.withMessage("The HttpMessageConverter list must not contain null elements");
|
||||
}
|
||||
|
||||
@Test
|
||||
public void setMessageConvertersPreconditions() {
|
||||
assertThatIllegalArgumentException()
|
||||
.isThrownBy(() -> template.setMessageConverters((List<HttpMessageConverter<?>>) null))
|
||||
.withMessage("At least one HttpMessageConverter is required");
|
||||
assertThatIllegalArgumentException()
|
||||
.isThrownBy(() -> template.setMessageConverters(Arrays.asList(null, this.converter)))
|
||||
.withMessage("The HttpMessageConverter list must not contain null elements");
|
||||
}
|
||||
|
||||
@Test
|
||||
public void varArgsTemplateVariables() throws Exception {
|
||||
|
||||
Reference in New Issue
Block a user