introspect element type in case of incoming Collection/Map in order to not accidentally say canConvert=true (SPR-6564)
This commit is contained in:
parent
5f9f69958e
commit
2153b2fbd5
|
|
@ -194,7 +194,7 @@ class TypeConverterDelegate {
|
|||
// No custom editor but custom ConversionService specified?
|
||||
ConversionService conversionService = this.propertyEditorRegistry.getConversionService();
|
||||
if (editor == null && conversionService != null && convertedValue != null) {
|
||||
TypeDescriptor sourceTypeDesc = TypeDescriptor.valueOf(convertedValue.getClass());
|
||||
TypeDescriptor sourceTypeDesc = new TypeDescriptor(convertedValue);
|
||||
if (conversionService.canConvert(sourceTypeDesc, typeDescriptor)) {
|
||||
return (T) conversionService.convert(convertedValue, sourceTypeDesc, typeDescriptor);
|
||||
}
|
||||
|
|
|
|||
|
|
@ -1,5 +1,24 @@
|
|||
/*
|
||||
* Copyright 2002-2006 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.context.conversionservice;
|
||||
|
||||
/**
|
||||
* @author Keith Donald
|
||||
*/
|
||||
public class Bar {
|
||||
|
||||
private String value;
|
||||
|
|
@ -11,4 +30,5 @@ public class Bar {
|
|||
public String getValue() {
|
||||
return value;
|
||||
}
|
||||
|
||||
}
|
||||
|
|
|
|||
|
|
@ -1,16 +1,32 @@
|
|||
/*
|
||||
* Copyright 2002-2006 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.context.conversionservice;
|
||||
|
||||
import static org.junit.Assert.assertEquals;
|
||||
import static org.junit.Assert.assertTrue;
|
||||
|
||||
import org.junit.Ignore;
|
||||
import static org.junit.Assert.*;
|
||||
import org.junit.Test;
|
||||
|
||||
import org.springframework.context.support.ClassPathXmlApplicationContext;
|
||||
|
||||
/**
|
||||
* @author Keith Donald
|
||||
*/
|
||||
public class ConversionServiceContextConfigTests {
|
||||
|
||||
@Test
|
||||
@Ignore
|
||||
public void testConfigOk() {
|
||||
ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("org/springframework/context/conversionservice/conversionService.xml");
|
||||
TestClient client = context.getBean("testClient", TestClient.class);
|
||||
|
|
|
|||
|
|
@ -1,7 +1,26 @@
|
|||
/*
|
||||
* Copyright 2002-2006 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.context.conversionservice;
|
||||
|
||||
import org.springframework.core.convert.converter.Converter;
|
||||
|
||||
/**
|
||||
* @author Keith Donald
|
||||
*/
|
||||
public class StringToBarConverter implements Converter<String, Bar> {
|
||||
|
||||
public Bar convert(String source) {
|
||||
|
|
|
|||
|
|
@ -1,15 +1,44 @@
|
|||
/*
|
||||
* Copyright 2002-2006 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.context.conversionservice;
|
||||
|
||||
import java.util.List;
|
||||
import java.util.Map;
|
||||
|
||||
import org.springframework.beans.factory.annotation.Autowired;
|
||||
import org.springframework.core.io.Resource;
|
||||
|
||||
/**
|
||||
* @author Keith Donald
|
||||
* @author Juergen Hoeller
|
||||
*/
|
||||
public class TestClient {
|
||||
|
||||
private List<Bar> bars;
|
||||
|
||||
private boolean bool;
|
||||
|
||||
private Resource[] resourceArray;
|
||||
|
||||
private List<Resource> resourceList;
|
||||
|
||||
private Map<String, Resource> resourceMap;
|
||||
|
||||
|
||||
public List<Bar> getBars() {
|
||||
return bars;
|
||||
}
|
||||
|
|
@ -27,4 +56,28 @@ public class TestClient {
|
|||
this.bool = bool;
|
||||
}
|
||||
|
||||
public Resource[] getResourceArray() {
|
||||
return resourceArray;
|
||||
}
|
||||
|
||||
public void setResourceArray(Resource[] resourceArray) {
|
||||
this.resourceArray = resourceArray;
|
||||
}
|
||||
|
||||
public List<Resource> getResourceList() {
|
||||
return resourceList;
|
||||
}
|
||||
|
||||
public void setResourceList(List<Resource> resourceList) {
|
||||
this.resourceList = resourceList;
|
||||
}
|
||||
|
||||
public Map<String, Resource> getResourceMap() {
|
||||
return resourceMap;
|
||||
}
|
||||
|
||||
public void setResourceMap(Map<String, Resource> resourceMap) {
|
||||
this.resourceMap = resourceMap;
|
||||
}
|
||||
|
||||
}
|
||||
|
|
|
|||
|
|
@ -5,7 +5,7 @@
|
|||
http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-3.0.xsd
|
||||
http://www.springframework.org/schema/util http://www.springframework.org/schema/util/spring-util-3.0.xsd">
|
||||
|
||||
<bean id="conversionService" class="org.springframework.core.convert.support.DefaultConversionService">
|
||||
<bean id="conversionService" class="org.springframework.context.support.ConversionServiceFactoryBean">
|
||||
<property name="converters">
|
||||
<bean class="org.springframework.context.conversionservice.StringToBarConverter" />
|
||||
</property>
|
||||
|
|
@ -13,6 +13,20 @@
|
|||
|
||||
<bean id="testClient" class="org.springframework.context.conversionservice.TestClient">
|
||||
<property name="bool" value="true"/>
|
||||
<property name="resourceArray">
|
||||
<value>classpath:test.xml</value>
|
||||
</property>
|
||||
<property name="resourceList">
|
||||
<list>
|
||||
<value>classpath:test.xml</value>
|
||||
</list>
|
||||
</property>
|
||||
<property name="resourceMap">
|
||||
<map>
|
||||
<entry key="res1" value="classpath:test1.xml"/>
|
||||
<entry key="res2" value="classpath:test2.xml"/>
|
||||
</map>
|
||||
</property>
|
||||
</bean>
|
||||
|
||||
<bean class="org.springframework.context.conversionservice.Bar">
|
||||
|
|
|
|||
|
|
@ -47,6 +47,8 @@ public class TypeDescriptor {
|
|||
public static final TypeDescriptor STRING = TypeDescriptor.valueOf(String.class);
|
||||
|
||||
|
||||
private Object value;
|
||||
|
||||
private Class<?> type;
|
||||
|
||||
private TypeDescriptor elementType;
|
||||
|
|
@ -58,10 +60,22 @@ public class TypeDescriptor {
|
|||
private Annotation[] cachedFieldAnnotations;
|
||||
|
||||
|
||||
/**
|
||||
* Create a new descriptor for the type of the given value.
|
||||
* <p>Use this constructor when a conversion point comes from a source such as a Map or
|
||||
* Collection, where no additional context is available but elements can be introspected.
|
||||
* @param type the actual type to wrap
|
||||
*/
|
||||
public TypeDescriptor(Object value) {
|
||||
Assert.notNull(value, "Value must not be null");
|
||||
this.value = value;
|
||||
this.type = value.getClass();
|
||||
}
|
||||
|
||||
/**
|
||||
* Create a new descriptor for the given type.
|
||||
* <p>Use this constructor when a conversion point comes from a source such as a Map
|
||||
* or Collection, where no additional context is available.
|
||||
* <p>Use this constructor when a conversion point comes from a plain source type,
|
||||
* where no additional context is available.
|
||||
* @param type the actual type to wrap
|
||||
*/
|
||||
public TypeDescriptor(Class<?> type) {
|
||||
|
|
@ -242,6 +256,7 @@ public class TypeDescriptor {
|
|||
* Determine the generic key type of the wrapped Map parameter/field, if any.
|
||||
* @return the generic type, or <code>null</code> if none
|
||||
*/
|
||||
@SuppressWarnings("unchecked")
|
||||
public Class<?> getMapKeyType() {
|
||||
if (this.field != null) {
|
||||
return GenericCollectionTypeResolver.getMapKeyFieldType(field);
|
||||
|
|
@ -249,6 +264,18 @@ public class TypeDescriptor {
|
|||
else if (this.methodParameter != null) {
|
||||
return GenericCollectionTypeResolver.getMapKeyParameterType(this.methodParameter);
|
||||
}
|
||||
else if (this.value instanceof Map) {
|
||||
Map map = (Map) this.value;
|
||||
if (!map.isEmpty()) {
|
||||
Object key = map.keySet().iterator().next();
|
||||
if (key != null) {
|
||||
return key.getClass();
|
||||
}
|
||||
}
|
||||
}
|
||||
if (this.type != null) {
|
||||
return GenericCollectionTypeResolver.getMapKeyType((Class<? extends Map>) this.type);
|
||||
}
|
||||
else {
|
||||
return null;
|
||||
}
|
||||
|
|
@ -258,6 +285,7 @@ public class TypeDescriptor {
|
|||
* Determine the generic value type of the wrapped Map parameter/field, if any.
|
||||
* @return the generic type, or <code>null</code> if none
|
||||
*/
|
||||
@SuppressWarnings("unchecked")
|
||||
public Class<?> getMapValueType() {
|
||||
if (this.field != null) {
|
||||
return GenericCollectionTypeResolver.getMapValueFieldType(this.field);
|
||||
|
|
@ -265,6 +293,18 @@ public class TypeDescriptor {
|
|||
else if (this.methodParameter != null) {
|
||||
return GenericCollectionTypeResolver.getMapValueParameterType(this.methodParameter);
|
||||
}
|
||||
else if (this.value instanceof Map) {
|
||||
Map map = (Map) this.value;
|
||||
if (!map.isEmpty()) {
|
||||
Object val = map.values().iterator().next();
|
||||
if (val != null) {
|
||||
return val.getClass();
|
||||
}
|
||||
}
|
||||
}
|
||||
if (this.type != null) {
|
||||
return GenericCollectionTypeResolver.getMapValueType((Class<? extends Map>) this.type);
|
||||
}
|
||||
else {
|
||||
return null;
|
||||
}
|
||||
|
|
@ -349,14 +389,26 @@ public class TypeDescriptor {
|
|||
|
||||
@SuppressWarnings("unchecked")
|
||||
private Class<?> getCollectionElementType() {
|
||||
if (this.field != null) {
|
||||
return GenericCollectionTypeResolver.getCollectionFieldType(this.field);
|
||||
}
|
||||
else if (this.methodParameter != null) {
|
||||
return GenericCollectionTypeResolver.getCollectionParameterType(this.methodParameter);
|
||||
}
|
||||
else if (this.value instanceof Collection) {
|
||||
Collection coll = (Collection) this.value;
|
||||
if (!coll.isEmpty()) {
|
||||
Object elem = coll.iterator().next();
|
||||
if (elem != null) {
|
||||
return elem.getClass();
|
||||
}
|
||||
}
|
||||
}
|
||||
if (this.type != null) {
|
||||
return GenericCollectionTypeResolver.getCollectionType((Class<? extends Collection>) this.type);
|
||||
}
|
||||
else if (this.field != null) {
|
||||
return GenericCollectionTypeResolver.getCollectionFieldType(this.field);
|
||||
}
|
||||
else {
|
||||
return GenericCollectionTypeResolver.getCollectionParameterType(this.methodParameter);
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
||||
|
|
|
|||
|
|
@ -1,15 +1,33 @@
|
|||
/*
|
||||
* Copyright 2002-2006 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.core.io.support;
|
||||
|
||||
import static org.junit.Assert.*;
|
||||
|
||||
import org.junit.Test;
|
||||
|
||||
import org.springframework.core.io.Resource;
|
||||
|
||||
|
||||
/**
|
||||
* @author Dave Syer
|
||||
*/
|
||||
public class ResourceArrayPropertyEditorTests {
|
||||
|
||||
|
||||
private ResourceArrayPropertyEditor editor = new ResourceArrayPropertyEditor();
|
||||
|
||||
|
||||
@Test
|
||||
public void testVanillaResource() throws Exception {
|
||||
editor.setAsText("classpath:org/springframework/core/io/support/ResourceArrayPropertyEditor.class");
|
||||
|
|
|
|||
Loading…
Reference in New Issue