@RequestBody raises 415 if no matching converter

This commit is contained in:
Rossen Stoyanchev 2016-06-07 22:14:59 -04:00
parent 36765f0f52
commit a8e5e40d97
4 changed files with 353 additions and 29 deletions

View File

@ -17,6 +17,9 @@
package org.springframework.web.reactive.result.method.annotation;
import java.util.List;
import java.util.stream.Collectors;
import javax.xml.crypto.Data;
import org.reactivestreams.Publisher;
import reactor.core.publisher.Flux;
@ -25,6 +28,7 @@ import reactor.core.publisher.Mono;
import org.springframework.core.MethodParameter;
import org.springframework.core.ResolvableType;
import org.springframework.core.convert.ConversionService;
import org.springframework.core.io.buffer.DataBuffer;
import org.springframework.http.MediaType;
import org.springframework.http.converter.reactive.HttpMessageConverter;
import org.springframework.ui.ModelMap;
@ -32,6 +36,7 @@ import org.springframework.util.Assert;
import org.springframework.web.bind.annotation.RequestBody;
import org.springframework.web.reactive.result.method.HandlerMethodArgumentResolver;
import org.springframework.web.server.ServerWebExchange;
import org.springframework.web.server.UnsupportedMediaTypeStatusException;
/**
* Resolves method arguments annotated with {@code @RequestBody} by reading and
@ -48,6 +53,8 @@ public class RequestBodyArgumentResolver implements HandlerMethodArgumentResolve
private final ConversionService conversionService;
private final List<MediaType> supportedMediaTypes;
/**
* Constructor with message converters and a ConversionService.
@ -61,6 +68,9 @@ public class RequestBodyArgumentResolver implements HandlerMethodArgumentResolve
Assert.notNull(service, "'conversionService' is required.");
this.messageConverters = converters;
this.conversionService = service;
this.supportedMediaTypes = converters.stream()
.flatMap(converter -> converter.getReadableMediaTypes().stream())
.collect(Collectors.toList());
}
@ -71,6 +81,13 @@ public class RequestBodyArgumentResolver implements HandlerMethodArgumentResolve
return this.messageConverters;
}
/**
* Return the configured {@link ConversionService}.
*/
public ConversionService getConversionService() {
return this.conversionService;
}
@Override
public boolean supportsParameter(MethodParameter parameter) {
@ -82,42 +99,40 @@ public class RequestBodyArgumentResolver implements HandlerMethodArgumentResolve
ServerWebExchange exchange) {
ResolvableType type = ResolvableType.forMethodParameter(parameter);
ResolvableType elementType = type.hasGenerics() ? type.getGeneric(0) : type;
boolean asyncType = isAsyncType(type);
ResolvableType elementType = (asyncType ? type.getGeneric(0) : type);
MediaType mediaType = exchange.getRequest().getHeaders().getContentType();
if (mediaType == null) {
mediaType = MediaType.APPLICATION_OCTET_STREAM;
}
Flux<?> elementFlux = exchange.getRequest().getBody();
HttpMessageConverter<?> converter = getMessageConverter(elementType, mediaType);
if (converter != null) {
elementFlux = converter.read(elementType, exchange.getRequest());
}
if (type.getRawClass() == Flux.class) {
return Mono.just(elementFlux);
}
else if (type.getRawClass() == Mono.class) {
return Mono.just(Mono.from(elementFlux));
}
else if (this.conversionService.canConvert(Publisher.class, type.getRawClass())) {
Object target = this.conversionService.convert(elementFlux, type.getRawClass());
return Mono.just(target);
}
// TODO Currently manage only "Foo" parameter, not "List<Foo>" parameters, Stéphane is going to add toIterable/toIterator to Flux to support that use case
return elementFlux.next().map(o -> o);
}
private HttpMessageConverter<?> getMessageConverter(ResolvableType type, MediaType mediaType) {
for (HttpMessageConverter<?> messageConverter : this.messageConverters) {
if (messageConverter.canRead(type, mediaType)) {
return messageConverter;
for (HttpMessageConverter<?> converter : getMessageConverters()) {
if (converter.canRead(elementType, mediaType)) {
Flux<?> elementFlux = converter.read(elementType, exchange.getRequest());
if (Mono.class.equals(type.getRawClass())) {
Object value = Mono.from(elementFlux);
return Mono.just(value);
}
else if (Flux.class.equals(type.getRawClass())) {
return Mono.just(elementFlux);
}
else if (asyncType) {
Object value = getConversionService().convert(elementFlux, type.getRawClass());
return Mono.just(value);
}
else {
// TODO Currently manage only "Foo" parameter, not "List<Foo>" parameters, Stéphane is going to add toIterable/toIterator to Flux to support that use case
return elementFlux.next().map(o -> o);
}
}
}
return null;
return Mono.error(new UnsupportedMediaTypeStatusException(mediaType, this.supportedMediaTypes));
}
private boolean isAsyncType(ResolvableType type) {
return (Mono.class.equals(type.getRawClass()) || Flux.class.equals(type.getRawClass()) ||
getConversionService().canConvert(Publisher.class, type.getRawClass()));
}
}

View File

@ -0,0 +1,59 @@
/*
* Copyright 2002-2016 the original author or authors.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.springframework.web.server;
import java.util.Collections;
import java.util.List;
import org.springframework.http.HttpStatus;
import org.springframework.http.MediaType;
/**
* Exception for errors that fit response status 415 (unsupported media type).
*
* @author Rossen Stoyanchev
*/
public class MediaTypeNotSupportedStatusException extends ResponseStatusException {
private final List<MediaType> supportedMediaTypes;
/**
* Constructor for when the Content-Type is invalid.
*/
public MediaTypeNotSupportedStatusException(String reason) {
super(HttpStatus.UNSUPPORTED_MEDIA_TYPE, reason);
this.supportedMediaTypes = Collections.emptyList();
}
/**
* Constructor for when the Content-Type is not supported.
*/
public MediaTypeNotSupportedStatusException(List<MediaType> supportedMediaTypes) {
super(HttpStatus.UNSUPPORTED_MEDIA_TYPE, "Unsupported media type", null);
this.supportedMediaTypes = Collections.unmodifiableList(supportedMediaTypes);
}
/**
* Return the list of supported content types in cases when the Accept
* header is parsed but not supported, or an empty list otherwise.
*/
public List<MediaType> getSupportedMediaTypes() {
return this.supportedMediaTypes;
}
}

View File

@ -68,4 +68,9 @@ public class Pojo {
public int hashCode() {
return 31 * foo.hashCode() + bar.hashCode();
}
@Override
public String toString() {
return "Pojo[foo='" + this.foo + "\'" + ", bar='" + this.bar + "\']";
}
}

View File

@ -0,0 +1,245 @@
/*
* Copyright 2002-2016 the original author or authors.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.springframework.web.reactive.result.method.annotation;
import java.lang.reflect.Method;
import java.net.URI;
import java.nio.ByteBuffer;
import java.nio.charset.Charset;
import java.time.Duration;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.concurrent.CompletableFuture;
import org.junit.Before;
import org.junit.Ignore;
import org.junit.Test;
import reactor.core.publisher.Flux;
import reactor.core.publisher.Mono;
import reactor.core.test.TestSubscriber;
import rx.Observable;
import rx.Single;
import org.springframework.core.LocalVariableTableParameterNameDiscoverer;
import org.springframework.core.MethodParameter;
import org.springframework.core.ParameterNameDiscoverer;
import org.springframework.core.annotation.SynthesizingMethodParameter;
import org.springframework.core.codec.Decoder;
import org.springframework.core.codec.support.JacksonJsonDecoder;
import org.springframework.core.codec.support.JsonObjectDecoder;
import org.springframework.core.codec.support.Pojo;
import org.springframework.core.codec.support.StringDecoder;
import org.springframework.core.convert.support.GenericConversionService;
import org.springframework.core.convert.support.ReactiveStreamsToCompletableFutureConverter;
import org.springframework.core.convert.support.ReactiveStreamsToRxJava1Converter;
import org.springframework.core.io.buffer.DataBuffer;
import org.springframework.core.io.buffer.DefaultDataBufferFactory;
import org.springframework.http.HttpMethod;
import org.springframework.http.MediaType;
import org.springframework.http.converter.reactive.CodecHttpMessageConverter;
import org.springframework.http.converter.reactive.HttpMessageConverter;
import org.springframework.http.server.reactive.MockServerHttpRequest;
import org.springframework.http.server.reactive.MockServerHttpResponse;
import org.springframework.ui.ExtendedModelMap;
import org.springframework.ui.ModelMap;
import org.springframework.util.ReflectionUtils;
import org.springframework.web.bind.annotation.RequestBody;
import org.springframework.web.server.ServerWebExchange;
import org.springframework.web.server.UnsupportedMediaTypeStatusException;
import org.springframework.web.server.adapter.DefaultServerWebExchange;
import org.springframework.web.server.session.DefaultWebSessionManager;
import static org.junit.Assert.assertArrayEquals;
import static org.junit.Assert.assertEquals;
import static org.junit.Assert.assertFalse;
import static org.junit.Assert.assertNotNull;
import static org.junit.Assert.assertTrue;
/**
* Unit tests for {@link RequestBodyArgumentResolver}.
* @author Rossen Stoyanchev
*/
public class RequestBodyArgumentResolverTests {
private RequestBodyArgumentResolver resolver;
private ServerWebExchange exchange;
private MockServerHttpRequest request;
private ModelMap model;
@Before
public void setUp() throws Exception {
this.resolver = resolver(new JacksonJsonDecoder(new JsonObjectDecoder()));
this.request = new MockServerHttpRequest(HttpMethod.GET, new URI("/path"));
MockServerHttpResponse response = new MockServerHttpResponse();
DefaultWebSessionManager sessionManager = new DefaultWebSessionManager();
this.exchange = new DefaultServerWebExchange(this.request, response, sessionManager);
this.model = new ExtendedModelMap();
}
@Test
public void supports() throws Exception {
RequestBodyArgumentResolver resolver = resolver(new StringDecoder());
assertTrue(resolver.supportsParameter(parameter("monoPojo")));
assertFalse(resolver.supportsParameter(parameter("paramWithoutAnnotation")));
}
@Test
public void missingContentType() throws Exception {
String body = "{\"bar\":\"BARBAR\",\"foo\":\"FOOFOO\"}";
this.request.writeWith(Flux.just(dataBuffer(body)));
Mono<Object> result = this.resolver.resolveArgument(parameter("monoPojo"), this.model, this.exchange);
TestSubscriber.subscribe(result)
.assertError(UnsupportedMediaTypeStatusException.class);
}
@Test @SuppressWarnings("unchecked")
public void monoPojo() throws Exception {
String body = "{\"bar\":\"b1\",\"foo\":\"f1\"}";
Mono<Pojo> mono = (Mono<Pojo>) resolve("monoPojo", Mono.class, body);
assertEquals(new Pojo("f1", "b1"), mono.block());
}
@Test @SuppressWarnings("unchecked")
public void fluxPojo() throws Exception {
String body = "[{\"bar\":\"b1\",\"foo\":\"f1\"},{\"bar\":\"b2\",\"foo\":\"f2\"}]";
Flux<Pojo> flux = (Flux<Pojo>) resolve("fluxPojo", Flux.class, body);
assertEquals(Arrays.asList(new Pojo("f1", "b1"), new Pojo("f2", "b2")), flux.collectList().block());
}
@Test @SuppressWarnings("unchecked")
public void singlePojo() throws Exception {
String body = "{\"bar\":\"b1\",\"foo\":\"f1\"}";
Single<Pojo> single = (Single<Pojo>) resolve("singlePojo", Single.class, body);
assertEquals(new Pojo("f1", "b1"), single.toBlocking().value());
}
@Test @SuppressWarnings("unchecked")
public void observablePojo() throws Exception {
String body = "[{\"bar\":\"b1\",\"foo\":\"f1\"},{\"bar\":\"b2\",\"foo\":\"f2\"}]";
Observable<?> observable = (Observable<?>) resolve("observablePojo", Observable.class, body);
assertEquals(Arrays.asList(new Pojo("f1", "b1"), new Pojo("f2", "b2")),
observable.toList().toBlocking().first());
}
@Test @SuppressWarnings("unchecked")
public void futurePojo() throws Exception {
String body = "{\"bar\":\"b1\",\"foo\":\"f1\"}";
assertEquals(new Pojo("f1", "b1"), resolve("futurePojo", CompletableFuture.class, body).get());
}
@Test
public void pojo() throws Exception {
String body = "{\"bar\":\"b1\",\"foo\":\"f1\"}";
assertEquals(new Pojo("f1", "b1"), resolve("pojo", Pojo.class, body));
}
@Test
public void map() throws Exception {
String body = "{\"bar\":\"b1\",\"foo\":\"f1\"}";
Map<String, String> map = new HashMap<>();
map.put("foo", "f1");
map.put("bar", "b1");
assertEquals(map, resolve("map", Map.class, body));
}
// TODO: @Ignore
@Test
@Ignore
public void list() throws Exception {
String body = "[{\"bar\":\"b1\",\"foo\":\"f1\"},{\"bar\":\"b2\",\"foo\":\"f2\"}]";
assertEquals(Arrays.asList(new Pojo("f1", "b1"), new Pojo("f2", "b2")),
resolve("list", List.class, body));
}
@Test
@Ignore
public void array() throws Exception {
String body = "[{\"bar\":\"b1\",\"foo\":\"f1\"},{\"bar\":\"b2\",\"foo\":\"f2\"}]";
assertArrayEquals(new Pojo[] {new Pojo("f1", "b1"), new Pojo("f2", "b2")},
resolve("array", Pojo[].class, body));
}
@SuppressWarnings("unchecked")
private <T> T resolve(String paramName, Class<T> valueType, String body) {
this.request.getHeaders().setContentType(MediaType.APPLICATION_JSON);
this.request.writeWith(Flux.just(dataBuffer(body)));
Mono<Object> result = this.resolver.resolveArgument(parameter(paramName), this.model, this.exchange);
Object value = result.block(Duration.ofSeconds(5));
assertNotNull(value);
assertTrue("Actual type: " + value.getClass(), valueType.isAssignableFrom(value.getClass()));
return (T) value;
}
@SuppressWarnings("Convert2MethodRef")
private RequestBodyArgumentResolver resolver(Decoder<?>... decoders) {
List<HttpMessageConverter<?>> converters = new ArrayList<>();
Arrays.asList(decoders).forEach(decoder -> converters.add(new CodecHttpMessageConverter<>(decoder)));
GenericConversionService service = new GenericConversionService();
service.addConverter(new ReactiveStreamsToCompletableFutureConverter());
service.addConverter(new ReactiveStreamsToRxJava1Converter());
return new RequestBodyArgumentResolver(converters, service);
}
@SuppressWarnings("ConfusingArgumentToVarargsMethod")
private MethodParameter parameter(String name) {
ParameterNameDiscoverer nameDiscoverer = new LocalVariableTableParameterNameDiscoverer();
Method method = ReflectionUtils.findMethod(getClass(), "handle", (Class<?>[]) null);
String[] names = nameDiscoverer.getParameterNames(method);
for (int i=0; i < names.length; i++) {
if (name.equals(names[i])) {
return new SynthesizingMethodParameter(method, i);
}
}
throw new IllegalArgumentException("Invalid parameter name '" + name + "'. Actual parameters: " +
Arrays.toString(names));
}
private DataBuffer dataBuffer(String body) {
byte[] bytes = body.getBytes(Charset.forName("UTF-8"));
ByteBuffer byteBuffer = ByteBuffer.wrap(bytes);
return new DefaultDataBufferFactory().wrap(byteBuffer);
}
@SuppressWarnings("unused")
void handle(
@RequestBody Mono<Pojo> monoPojo,
@RequestBody Flux<Pojo> fluxPojo,
@RequestBody Single<Pojo> singlePojo,
@RequestBody Observable<Pojo> observablePojo,
@RequestBody CompletableFuture<Pojo> futurePojo,
@RequestBody Pojo pojo,
@RequestBody Map<String, String> map,
@RequestBody List<Pojo> list,
@RequestBody Set<Pojo> set,
@RequestBody Pojo[] array,
Pojo paramWithoutAnnotation) {
}
}