From 3cd5e1a1dab61b8a6b328c0839211b6862d15521 Mon Sep 17 00:00:00 2001 From: Ilia Motornyi Date: Sat, 24 Mar 2018 11:57:26 +0200 Subject: Always return a fresh copy of BeanPropertySet from cache (#10635) --- .../main/java/com/vaadin/data/BeanPropertySet.java | 81 +++++++++-------- .../java/com/vaadin/data/BeanPropertySetTest.java | 57 +++++++----- .../com/vaadin/data/PropertyRetrospectionTest.java | 101 +++++++++++++++++++++ 3 files changed, 176 insertions(+), 63 deletions(-) create mode 100644 server/src/test/java/com/vaadin/data/PropertyRetrospectionTest.java diff --git a/server/src/main/java/com/vaadin/data/BeanPropertySet.java b/server/src/main/java/com/vaadin/data/BeanPropertySet.java index b877fdf939..6b234703e5 100644 --- a/server/src/main/java/com/vaadin/data/BeanPropertySet.java +++ b/server/src/main/java/com/vaadin/data/BeanPropertySet.java @@ -55,11 +55,11 @@ public class BeanPropertySet implements PropertySet { * @see #readResolve() * @see BeanPropertyDefinition#writeReplace() */ - private static class SerializedPropertySet implements Serializable { - private final Class beanType; + private static class SerializedPropertySet implements Serializable { + private final InstanceKey instanceKey; - private SerializedPropertySet(Class beanType) { - this.beanType = beanType; + private SerializedPropertySet(InstanceKey instanceKey) { + this.instanceKey = instanceKey; } private Object readResolve() { @@ -67,7 +67,8 @@ public class BeanPropertySet implements PropertySet { * When this instance is deserialized, it will be replaced with a * property set for the corresponding bean type and property name. */ - return get(beanType); + return get(instanceKey.type,instanceKey.checkNestedDefinitions, + new PropertyFilterDefinition(instanceKey.depth,instanceKey.ignorePackageNames)); } } @@ -143,7 +144,7 @@ public class BeanPropertySet implements PropertySet { * serialize a DTO that when deserialized will get the corresponding * property definition from the cache. */ - return new SerializedPropertyDefinition(getPropertySet().beanType, + return new SerializedPropertyDefinition(getPropertySet().instanceKey.type, getName()); } } @@ -221,7 +222,7 @@ public class BeanPropertySet implements PropertySet { * serialize a DTO that when deserialized will get the corresponding * property definition from the cache. */ - return new SerializedPropertyDefinition(getPropertySet().beanType, + return new SerializedPropertyDefinition(getPropertySet().instanceKey.type, getName()); } @@ -241,13 +242,13 @@ public class BeanPropertySet implements PropertySet { * * @since 8.2 */ - private static class InstanceKey implements Serializable { - private Class type; + private static class InstanceKey implements Serializable { + private Class type; private boolean checkNestedDefinitions; private int depth; private List ignorePackageNames; - public InstanceKey(Class type, boolean checkNestedDefinitions, + public InstanceKey(Class type, boolean checkNestedDefinitions, int depth, List ignorePackageNames) { this.type = type; this.checkNestedDefinitions = checkNestedDefinitions; @@ -292,45 +293,43 @@ public class BeanPropertySet implements PropertySet { } else if (!ignorePackageNames.equals(other.ignorePackageNames)) { return false; } - if (type == null) { - if (other.type != null) { - return false; - } - } else if (!type.equals(other.type)) { - return false; - } - return true; + return Objects.equals(type, other.type); } } - private static final ConcurrentMap> INSTANCES = new ConcurrentHashMap<>(); + private static final ConcurrentMap, BeanPropertySet> INSTANCES = new ConcurrentHashMap<>(); - private final Class beanType; + private final InstanceKey instanceKey; private final Map> definitions; - private BeanPropertySet(Class beanType) { - this.beanType = beanType; + private BeanPropertySet(InstanceKey instanceKey) { + this.instanceKey = instanceKey; try { - definitions = BeanUtil.getBeanPropertyDescriptors(beanType).stream() + definitions = BeanUtil.getBeanPropertyDescriptors(instanceKey.type).stream() .filter(BeanPropertySet::hasNonObjectReadMethod) .map(descriptor -> new BeanPropertyDefinition<>(this, - beanType, descriptor)) + instanceKey.type, descriptor)) .collect(Collectors.toMap(PropertyDefinition::getName, Function.identity())); } catch (IntrospectionException e) { throw new IllegalArgumentException( "Cannot find property descriptors for " - + beanType.getName(), + + instanceKey.type.getName(), e); } } - private BeanPropertySet(Class beanType, boolean checkNestedDefinitions, - PropertyFilterDefinition propertyFilterDefinition) { - this(beanType); + private BeanPropertySet(InstanceKey instanceKey, Map> definitions) { + this.instanceKey = instanceKey; + this.definitions = new HashMap<>(definitions); + } + + private BeanPropertySet(InstanceKey instanceKey, boolean checkNestedDefinitions, + PropertyFilterDefinition propertyFilterDefinition) { + this(instanceKey); if (checkNestedDefinitions) { Objects.requireNonNull(propertyFilterDefinition, "You must define a property filter callback if using nested property scan."); @@ -370,7 +369,7 @@ public class BeanPropertySet implements PropertySet { String name = parentPropertyKey + "." + descriptor.getName(); PropertyDescriptor subDescriptor = BeanUtil - .getPropertyDescriptor(beanType, name); + .getPropertyDescriptor(instanceKey.type, name); moreProps.put(name, new NestedBeanPropertyDefinition<>(this, parentProperty, subDescriptor)); @@ -402,7 +401,11 @@ public class BeanPropertySet implements PropertySet { InstanceKey key = new InstanceKey(beanType, false, 0, null); // Cache the reflection results return (PropertySet) INSTANCES.computeIfAbsent(key, - ignored -> new BeanPropertySet<>(beanType)); + ignored -> new BeanPropertySet<>(key)).copy(); + } + + private BeanPropertySet copy() { + return new BeanPropertySet<>(instanceKey,definitions); } /** @@ -426,8 +429,8 @@ public class BeanPropertySet implements PropertySet { filterDefinition.getMaxNestingDepth(), filterDefinition.getIgnorePackageNamesStartingWith()); return (PropertySet) INSTANCES.computeIfAbsent(key, - k -> new BeanPropertySet<>(beanType, checkNestedDefinitions, - filterDefinition)); + k -> new BeanPropertySet<>(key, checkNestedDefinitions, + filterDefinition)).copy(); } @Override @@ -448,11 +451,11 @@ public class BeanPropertySet implements PropertySet { if (!parent.isPresent()) { throw new IllegalArgumentException( "Cannot find property descriptor [" + parentName - + "] for " + beanType.getName()); + + "] for " + instanceKey.type.getName()); } Optional descriptor = Optional.ofNullable( - BeanUtil.getPropertyDescriptor(beanType, name)); + BeanUtil.getPropertyDescriptor(instanceKey.type, name)); if (descriptor.isPresent()) { NestedBeanPropertyDefinition nestedDefinition = new NestedBeanPropertyDefinition<>( this, parent.get(), descriptor.get()); @@ -461,13 +464,13 @@ public class BeanPropertySet implements PropertySet { } else { throw new IllegalArgumentException( "Cannot find property descriptor [" + name - + "] for " + beanType.getName()); + + "] for " + instanceKey.type.getName()); } } catch (IntrospectionException e) { throw new IllegalArgumentException( "Cannot find property descriptors for " - + beanType.getName(), + + instanceKey.type.getName(), e); } } @@ -481,7 +484,7 @@ public class BeanPropertySet implements PropertySet { * @return the bean type of this bean property set */ public Class getBeanType() { - return beanType; + return instanceKey.type; } private static boolean hasNonObjectReadMethod( @@ -502,7 +505,7 @@ public class BeanPropertySet implements PropertySet { @Override public String toString() { - return "Property set for bean " + beanType.getName(); + return "Property set for bean " + instanceKey.type.getName(); } private Object writeReplace() { @@ -511,6 +514,6 @@ public class BeanPropertySet implements PropertySet { * that when deserialized will get the corresponding property set from * the cache. */ - return new SerializedPropertySet(beanType); + return new SerializedPropertySet(instanceKey); } } diff --git a/server/src/test/java/com/vaadin/data/BeanPropertySetTest.java b/server/src/test/java/com/vaadin/data/BeanPropertySetTest.java index 6b43bdfa24..d319a1c52a 100644 --- a/server/src/test/java/com/vaadin/data/BeanPropertySetTest.java +++ b/server/src/test/java/com/vaadin/data/BeanPropertySetTest.java @@ -15,21 +15,19 @@ */ package com.vaadin.data; -import static org.junit.Assert.assertEquals; -import static org.junit.Assert.assertNotSame; -import static org.junit.Assert.assertSame; - import java.io.ByteArrayInputStream; import java.io.ByteArrayOutputStream; import java.io.ObjectInputStream; import java.io.ObjectOutputStream; import java.lang.reflect.Field; import java.util.Arrays; +import java.util.Comparator; import java.util.HashSet; import java.util.Map; import java.util.Set; import java.util.stream.Collectors; +import junit.framework.AssertionFailedError; import org.junit.Test; import com.vaadin.data.provider.bov.Person; @@ -39,6 +37,11 @@ import com.vaadin.tests.data.bean.FatherAndSon; import com.vaadin.tests.data.bean.Sex; import com.vaadin.tests.server.ClassesSerializableTest; +import static com.vaadin.data.PropertyFilterDefinition.getDefaultFilter; +import static org.junit.Assert.assertEquals; +import static org.junit.Assert.assertSame; +import static org.junit.Assert.assertNotSame; + public class BeanPropertySetTest { @Test public void testSerializeDeserialize_propertySet() throws Exception { @@ -48,8 +51,23 @@ public class BeanPropertySetTest { PropertySet deserializedPropertySet = ClassesSerializableTest .serializeAndDeserialize(originalPropertySet); - assertSame("Deserialized instance should be the same as the original", - originalPropertySet, deserializedPropertySet); + comparePropertySet(originalPropertySet, deserializedPropertySet, "Deserialized instance should be the same as the original"); + } + + private void comparePropertySet(PropertySet propertySetA, PropertySet propertySetB, String message) { + + PropertyDefinition[] propertiesA = propertySetA.getProperties() + .sorted(Comparator.comparing(PropertyDefinition::getName)) + .toArray(PropertyDefinition[]::new); + PropertyDefinition[] propertiesB = propertySetA.getProperties() + .sorted(Comparator.comparing(PropertyDefinition::getName)) + .toArray(PropertyDefinition[]::new); + + assertEquals(message, propertiesA.length, propertiesB.length); + for (int i = 0; i < propertiesB.length; i++) { + assertSame(message, + propertiesA[i], propertiesB[i]); + } } @Test @@ -76,8 +94,7 @@ public class BeanPropertySetTest { PropertySet deserializedPropertySet = (PropertySet) in .readObject(); - assertSame("Deserialized instance should be the same as in the cache", - BeanPropertySet.get(Person.class), deserializedPropertySet); + comparePropertySet(BeanPropertySet.get(Person.class), deserializedPropertySet, "Deserialized instance should be the same as in the cache"); assertNotSame( "Deserialized instance should not be the same as the original", originalPropertySet, deserializedPropertySet); @@ -108,10 +125,11 @@ public class BeanPropertySetTest { @Test public void testSerializeDeserialize_nestedPropertyDefinition() throws Exception { + PropertyDefinition definition = BeanPropertySet - .get(com.vaadin.tests.data.bean.Person.class) - .getProperty("address.postalCode") - .orElseThrow(RuntimeException::new); + .get(com.vaadin.tests.data.bean.Person.class,true, getDefaultFilter()) + .getProperty("address.postalCode").orElseThrow(AssertionFailedError::new); + PropertyDefinition deserializedDefinition = ClassesSerializableTest .serializeAndDeserialize(definition); @@ -128,16 +146,10 @@ public class BeanPropertySetTest { assertEquals("Deserialized definition should be functional", address.getPostalCode(), postalCode); - assertSame("Deserialized instance should be the same as in the cache", - BeanPropertySet.get(com.vaadin.tests.data.bean.Person.class) - .getProperty("address.postalCode").orElseThrow( - RuntimeException::new), - deserializedDefinition); } @Test - public void nestedPropertyDefinition_samePropertyNameOnMultipleLevels() - throws Exception { + public void nestedPropertyDefinition_samePropertyNameOnMultipleLevels() { PropertyDefinition definition = BeanPropertySet .get(FatherAndSon.class).getProperty("father.father.firstName") .orElseThrow(RuntimeException::new); @@ -156,8 +168,7 @@ public class BeanPropertySetTest { } @Test(expected = NullPointerException.class) - public void nestedPropertyDefinition_propertyChainBroken() - throws Exception { + public void nestedPropertyDefinition_propertyChainBroken() { PropertyDefinition definition = BeanPropertySet .get(FatherAndSon.class).getProperty("father.firstName") .orElseThrow(RuntimeException::new); @@ -166,15 +177,13 @@ public class BeanPropertySetTest { } @Test(expected = IllegalArgumentException.class) - public void nestedPropertyDefinition_invalidPropertyNameInChain() - throws Exception { + public void nestedPropertyDefinition_invalidPropertyNameInChain() { BeanPropertySet.get(FatherAndSon.class) .getProperty("grandfather.firstName"); } @Test(expected = IllegalArgumentException.class) - public void nestedPropertyDefinition_invalidPropertyNameAtChainEnd() - throws Exception { + public void nestedPropertyDefinition_invalidPropertyNameAtChainEnd() { BeanPropertySet.get(FatherAndSon.class).getProperty("father.age"); } diff --git a/server/src/test/java/com/vaadin/data/PropertyRetrospectionTest.java b/server/src/test/java/com/vaadin/data/PropertyRetrospectionTest.java new file mode 100644 index 0000000000..3b3a611345 --- /dev/null +++ b/server/src/test/java/com/vaadin/data/PropertyRetrospectionTest.java @@ -0,0 +1,101 @@ +package com.vaadin.data; + +import org.junit.Test; + +import com.vaadin.ui.Grid; +import com.vaadin.ui.TextField; + +import static org.junit.Assert.assertEquals; + +public class PropertyRetrospectionTest { + + @SuppressWarnings("unused") + public static class InnerBean { + private String innerString; + + public String getInnerString() { + return innerString; + } + + public void setInnerString(String innerString) { + this.innerString = innerString; + } + } + + @SuppressWarnings("unused") + public static class BeanOne { + private String someString; + private InnerBean innerBean; + + public String getSomeString() { + return someString; + } + + public void setSomeString(String someString) { + this.someString = someString; + } + + public InnerBean getInnerBean() { + return innerBean; + } + + public void setInnerBean(InnerBean innerBean) { + this.innerBean = innerBean; + } + } + + @SuppressWarnings("unused") + public static class BeanTwo { + private String someString; + private InnerBean innerBean; + + public String getSomeString() { + return someString; + } + + public void setSomeString(String someString) { + this.someString = someString; + } + + public InnerBean getInnerBean() { + return innerBean; + } + + public void setInnerBean(InnerBean innerBean) { + this.innerBean = innerBean; + } + } + + @Test + public void testGridBeanProperties() + { + Grid grid1 = new Grid<>(BeanOne.class); + assertEquals(2,BeanPropertySet.get(BeanOne.class).getProperties().count()); + assertEquals(2,grid1.getColumns().size()); + grid1.addColumn("innerBean.innerString"); + assertEquals(3,grid1.getColumns().size()); + assertEquals(2,BeanPropertySet.get(BeanOne.class).getProperties().count()); + + + Grid grid2 = new Grid<>(BeanOne.class); + assertEquals(2,BeanPropertySet.get(BeanOne.class).getProperties().count()); + assertEquals(2,grid2.getColumns().size()); + grid2.addColumn("innerBean.innerString"); + assertEquals(3,grid2.getColumns().size()); + assertEquals(2,BeanPropertySet.get(BeanOne.class).getProperties().count()); + } + + @Test + public void testBinder() + { + Binder binder1 = new Binder<>(BeanTwo.class); + assertEquals(2,BeanPropertySet.get(BeanTwo.class).getProperties().count()); + binder1.forField(new TextField()).bind("innerBean.innerString"); + assertEquals(2,BeanPropertySet.get(BeanTwo.class).getProperties().count()); + + Binder binder2 = new Binder<>(BeanTwo.class); + assertEquals(2,BeanPropertySet.get(BeanTwo.class).getProperties().count()); + binder2.forField(new TextField()).bind("innerBean.innerString"); + assertEquals(2,BeanPropertySet.get(BeanTwo.class).getProperties().count()); + } +} -- cgit v1.2.3