SPR-6686 - @ResponseBody throws HttpMediaTypeNotAcceptableException if client accepts "*/*"
git-svn-id: https://src.springframework.org/svn/spring-framework/trunk@2814 50f2f4bb-b051-0410-bef5-90022cba6387
This commit is contained in:
parent
b695f63a0b
commit
c3ab0ef473
|
|
@ -864,6 +864,9 @@ public class AnnotationMethodHandlerAdapter extends WebContentGenerator
|
||||||
if (acceptedMediaTypes.isEmpty()) {
|
if (acceptedMediaTypes.isEmpty()) {
|
||||||
acceptedMediaTypes = Collections.singletonList(MediaType.ALL);
|
acceptedMediaTypes = Collections.singletonList(MediaType.ALL);
|
||||||
}
|
}
|
||||||
|
else {
|
||||||
|
Collections.sort(acceptedMediaTypes);
|
||||||
|
}
|
||||||
HttpOutputMessage outputMessage = new ServletServerHttpResponse(webRequest.getResponse());
|
HttpOutputMessage outputMessage = new ServletServerHttpResponse(webRequest.getResponse());
|
||||||
Class<?> returnValueType = returnValue.getClass();
|
Class<?> returnValueType = returnValue.getClass();
|
||||||
List<MediaType> allSupportedMediaTypes = new ArrayList<MediaType>();
|
List<MediaType> allSupportedMediaTypes = new ArrayList<MediaType>();
|
||||||
|
|
@ -872,7 +875,7 @@ public class AnnotationMethodHandlerAdapter extends WebContentGenerator
|
||||||
allSupportedMediaTypes.addAll(messageConverter.getSupportedMediaTypes());
|
allSupportedMediaTypes.addAll(messageConverter.getSupportedMediaTypes());
|
||||||
for (MediaType acceptedMediaType : acceptedMediaTypes) {
|
for (MediaType acceptedMediaType : acceptedMediaTypes) {
|
||||||
if (messageConverter.canWrite(returnValueType, acceptedMediaType)) {
|
if (messageConverter.canWrite(returnValueType, acceptedMediaType)) {
|
||||||
messageConverter.write(returnValue, null, outputMessage);
|
messageConverter.write(returnValue, acceptedMediaType, outputMessage);
|
||||||
this.responseArgumentUsed = true;
|
this.responseArgumentUsed = true;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -92,7 +92,26 @@ public abstract class AbstractHttpMessageConverter<T> implements HttpMessageConv
|
||||||
* type.
|
* type.
|
||||||
*/
|
*/
|
||||||
public boolean canRead(Class<? extends T> clazz, MediaType mediaType) {
|
public boolean canRead(Class<? extends T> clazz, MediaType mediaType) {
|
||||||
return supports(clazz) && isSupported(mediaType);
|
return supports(clazz) && canRead(mediaType);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns true if any of the {@linkplain #setSupportedMediaTypes(List) supported media types} include the given media
|
||||||
|
* type.
|
||||||
|
*
|
||||||
|
* @param mediaType the media type
|
||||||
|
* @return true if the supported media types include the media type, or if the media type is {@code null}
|
||||||
|
*/
|
||||||
|
protected boolean canRead(MediaType mediaType) {
|
||||||
|
if (mediaType == null) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
for (MediaType supportedMediaType : getSupportedMediaTypes()) {
|
||||||
|
if (supportedMediaType.includes(mediaType)) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
@ -103,22 +122,22 @@ public abstract class AbstractHttpMessageConverter<T> implements HttpMessageConv
|
||||||
* type.
|
* type.
|
||||||
*/
|
*/
|
||||||
public boolean canWrite(Class<? extends T> clazz, MediaType mediaType) {
|
public boolean canWrite(Class<? extends T> clazz, MediaType mediaType) {
|
||||||
return supports(clazz) && isSupported(mediaType);
|
return supports(clazz) && canWrite(mediaType);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns true if any of the {@linkplain #setSupportedMediaTypes(List) supported media types} include the given media
|
* Returns true if the given media type includes any of the
|
||||||
* type.
|
* {@linkplain #setSupportedMediaTypes(List) supported media types}.
|
||||||
*
|
*
|
||||||
* @param mediaType the media type
|
* @param mediaType the media type
|
||||||
* @return true if the supported media types include the media type, or if the media type is {@code null}
|
* @return true if the supported media types include the media type, or if the media type is {@code null}
|
||||||
*/
|
*/
|
||||||
protected boolean isSupported(MediaType mediaType) {
|
protected boolean canWrite(MediaType mediaType) {
|
||||||
if (mediaType == null) {
|
if (mediaType == null) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
for (MediaType supportedMediaType : getSupportedMediaTypes()) {
|
for (MediaType supportedMediaType : getSupportedMediaTypes()) {
|
||||||
if (supportedMediaType.includes(mediaType)) {
|
if (mediaType.includes(supportedMediaType)) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
@ -165,7 +184,7 @@ public abstract class AbstractHttpMessageConverter<T> implements HttpMessageConv
|
||||||
public final void write(T t, MediaType contentType, HttpOutputMessage outputMessage)
|
public final void write(T t, MediaType contentType, HttpOutputMessage outputMessage)
|
||||||
throws IOException, HttpMessageNotWritableException {
|
throws IOException, HttpMessageNotWritableException {
|
||||||
HttpHeaders headers = outputMessage.getHeaders();
|
HttpHeaders headers = outputMessage.getHeaders();
|
||||||
if (contentType == null) {
|
if (contentType == null || contentType.isWildcardType() || contentType.isWildcardSubtype()) {
|
||||||
contentType = getDefaultContentType(t);
|
contentType = getDefaultContentType(t);
|
||||||
}
|
}
|
||||||
if (contentType != null) {
|
if (contentType != null) {
|
||||||
|
|
|
||||||
|
|
@ -89,12 +89,12 @@ public class MappingJacksonHttpMessageConverter extends AbstractHttpMessageConve
|
||||||
@Override
|
@Override
|
||||||
public boolean canRead(Class<?> clazz, MediaType mediaType) {
|
public boolean canRead(Class<?> clazz, MediaType mediaType) {
|
||||||
JavaType javaType = TypeFactory.fromClass(clazz);
|
JavaType javaType = TypeFactory.fromClass(clazz);
|
||||||
return objectMapper.canDeserialize(javaType) && isSupported(mediaType);
|
return objectMapper.canDeserialize(javaType) && canRead(mediaType);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean canWrite(Class<?> clazz, MediaType mediaType) {
|
public boolean canWrite(Class<?> clazz, MediaType mediaType) {
|
||||||
return objectMapper.canSerialize(clazz) && isSupported(mediaType);
|
return objectMapper.canSerialize(clazz) && canWrite(mediaType);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
|
|
||||||
|
|
@ -52,12 +52,12 @@ public class Jaxb2RootElementHttpMessageConverter extends AbstractJaxb2HttpMessa
|
||||||
@Override
|
@Override
|
||||||
public boolean canRead(Class<?> clazz, MediaType mediaType) {
|
public boolean canRead(Class<?> clazz, MediaType mediaType) {
|
||||||
return (clazz.isAnnotationPresent(XmlRootElement.class) || clazz.isAnnotationPresent(XmlType.class)) &&
|
return (clazz.isAnnotationPresent(XmlRootElement.class) || clazz.isAnnotationPresent(XmlType.class)) &&
|
||||||
isSupported(mediaType);
|
canRead(mediaType);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean canWrite(Class<?> clazz, MediaType mediaType) {
|
public boolean canWrite(Class<?> clazz, MediaType mediaType) {
|
||||||
return AnnotationUtils.findAnnotation(clazz, XmlRootElement.class) != null && isSupported(mediaType);
|
return AnnotationUtils.findAnnotation(clazz, XmlRootElement.class) != null && canWrite(mediaType);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
|
|
||||||
|
|
@ -36,6 +36,17 @@ public class ByteArrayHttpMessageConverterTests {
|
||||||
converter = new ByteArrayHttpMessageConverter();
|
converter = new ByteArrayHttpMessageConverter();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void canRead() {
|
||||||
|
assertTrue(converter.canRead(byte[].class, new MediaType("application", "octet-stream")));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void canWrite() {
|
||||||
|
assertTrue(converter.canWrite(byte[].class, new MediaType("application", "octet-stream")));
|
||||||
|
assertTrue(converter.canWrite(byte[].class, MediaType.ALL));
|
||||||
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void read() throws IOException {
|
public void read() throws IOException {
|
||||||
byte[] body = new byte[]{0x1, 0x2};
|
byte[] body = new byte[]{0x1, 0x2};
|
||||||
|
|
|
||||||
|
|
@ -40,6 +40,19 @@ public class FormHttpMessageConverterTests {
|
||||||
converter = new FormHttpMessageConverter();
|
converter = new FormHttpMessageConverter();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
@SuppressWarnings("unchecked")
|
||||||
|
public void canRead() {
|
||||||
|
assertTrue(converter.canRead((Class<? extends MultiValueMap<String, String>>) MultiValueMap.class, new MediaType("application", "x-www-form-urlencoded")));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
@SuppressWarnings("unchecked")
|
||||||
|
public void canWrite() {
|
||||||
|
assertTrue(converter.canWrite((Class<? extends MultiValueMap<String, String>>) MultiValueMap.class, new MediaType("application", "x-www-form-urlencoded")));
|
||||||
|
assertTrue(converter.canWrite((Class<? extends MultiValueMap<String, String>>) MultiValueMap.class, MediaType.ALL));
|
||||||
|
}
|
||||||
|
|
||||||
@SuppressWarnings("unchecked")
|
@SuppressWarnings("unchecked")
|
||||||
@Test
|
@Test
|
||||||
public void read() throws Exception {
|
public void read() throws Exception {
|
||||||
|
|
|
||||||
|
|
@ -0,0 +1,98 @@
|
||||||
|
/*
|
||||||
|
* Copyright 2002-2010 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.http.converter;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
|
||||||
|
import org.junit.Before;
|
||||||
|
import org.junit.Test;
|
||||||
|
import static org.junit.Assert.*;
|
||||||
|
|
||||||
|
import org.springframework.http.HttpInputMessage;
|
||||||
|
import org.springframework.http.HttpOutputMessage;
|
||||||
|
import org.springframework.http.MediaType;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Test-case for AbstractHttpMessageConverter.
|
||||||
|
*
|
||||||
|
* @author Arjen Poutsma
|
||||||
|
*/
|
||||||
|
public class HttpMessageConverterTests {
|
||||||
|
|
||||||
|
private static final MediaType MEDIA_TYPE = new MediaType("foo", "bar");
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void canRead() {
|
||||||
|
AbstractHttpMessageConverter<MyType> converter = new MyHttpMessageConverter<MyType>(MEDIA_TYPE) {
|
||||||
|
@Override
|
||||||
|
protected boolean supports(Class<? extends MyType> clazz) {
|
||||||
|
return MyType.class.equals(clazz);
|
||||||
|
}
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
assertTrue(converter.canRead(MyType.class, MEDIA_TYPE));
|
||||||
|
assertFalse(converter.canRead(MyType.class, new MediaType("foo", "*")));
|
||||||
|
assertFalse(converter.canRead(MyType.class, MediaType.ALL));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void canWrite() {
|
||||||
|
AbstractHttpMessageConverter<MyType> converter = new MyHttpMessageConverter<MyType>(MEDIA_TYPE) {
|
||||||
|
@Override
|
||||||
|
protected boolean supports(Class<? extends MyType> clazz) {
|
||||||
|
return MyType.class.equals(clazz);
|
||||||
|
}
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
assertTrue(converter.canWrite(MyType.class, MEDIA_TYPE));
|
||||||
|
assertTrue(converter.canWrite(MyType.class, new MediaType("foo", "*")));
|
||||||
|
assertTrue(converter.canWrite(MyType.class, MediaType.ALL));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
private static class MyHttpMessageConverter<T> extends AbstractHttpMessageConverter<T> {
|
||||||
|
|
||||||
|
private MyHttpMessageConverter(MediaType supportedMediaType) {
|
||||||
|
super(supportedMediaType);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected boolean supports(Class<? extends T> clazz) {
|
||||||
|
fail("Not expected");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected T readInternal(Class<T> clazz, HttpInputMessage inputMessage)
|
||||||
|
throws IOException, HttpMessageNotReadableException {
|
||||||
|
fail("Not expected");
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void writeInternal(T t, HttpOutputMessage outputMessage)
|
||||||
|
throws IOException, HttpMessageNotWritableException {
|
||||||
|
fail("Not expected");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static class MyType {
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
@ -37,6 +37,17 @@ public class StringHttpMessageConverterTests {
|
||||||
converter = new StringHttpMessageConverter();
|
converter = new StringHttpMessageConverter();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void canRead() {
|
||||||
|
assertTrue(converter.canRead(String.class, new MediaType("text", "plain")));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void canWrite() {
|
||||||
|
assertTrue(converter.canWrite(String.class, new MediaType("text", "plain")));
|
||||||
|
assertTrue(converter.canWrite(String.class, MediaType.ALL));
|
||||||
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void read() throws IOException {
|
public void read() throws IOException {
|
||||||
String body = "Hello World";
|
String body = "Hello World";
|
||||||
|
|
|
||||||
Loading…
Reference in New Issue