OPENJPA-2767: Incomplete ValueMapDiscriminatorStrategy cache and MetaDataRepository race condition

Signed-off-by: Will Dazey <dazeydev.3@gmail.com>
This commit is contained in:
Will Dazey 2019-05-14 15:25:43 -05:00
parent 9f26ed29bf
commit 389a82db97
2 changed files with 81 additions and 37 deletions

View File

@ -24,6 +24,8 @@ import java.util.HashMap;
import java.util.List; import java.util.List;
import java.util.Map; import java.util.Map;
import java.util.TreeSet; import java.util.TreeSet;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantReadWriteLock;
import org.apache.openjpa.jdbc.kernel.JDBCStore; import org.apache.openjpa.jdbc.kernel.JDBCStore;
import org.apache.openjpa.jdbc.meta.ClassMapping; import org.apache.openjpa.jdbc.meta.ClassMapping;
@ -43,7 +45,6 @@ import org.apache.openjpa.util.MetaDataException;
public class ValueMapDiscriminatorStrategy public class ValueMapDiscriminatorStrategy
extends InValueDiscriminatorStrategy { extends InValueDiscriminatorStrategy {
private static final long serialVersionUID = 1L; private static final long serialVersionUID = 1L;
public static final String ALIAS = "value-map"; public static final String ALIAS = "value-map";
@ -51,7 +52,10 @@ public class ValueMapDiscriminatorStrategy
private static final Localizer _loc = Localizer.forPackage private static final Localizer _loc = Localizer.forPackage
(ValueMapDiscriminatorStrategy.class); (ValueMapDiscriminatorStrategy.class);
private Map _vals = null; private Map<String, Class<?>> _vals;
private final ReentrantReadWriteLock rwl = new ReentrantReadWriteLock(true);
private final Lock _readLock = rwl.readLock();
private final Lock _writeLock = rwl.writeLock();
@Override @Override
public String getAlias() { public String getAlias() {
@ -67,8 +71,8 @@ public class ValueMapDiscriminatorStrategy
// if the user wants the type to be null, we need a jdbc-type // if the user wants the type to be null, we need a jdbc-type
// on the column or an existing column to figure out the java type // on the column or an existing column to figure out the java type
DiscriminatorMappingInfo info = disc.getMappingInfo(); DiscriminatorMappingInfo info = disc.getMappingInfo();
List cols = info.getColumns(); List<Column> cols = info.getColumns();
Column col = (cols.isEmpty()) ? null : (Column) cols.get(0); Column col = (cols.isEmpty()) ? null : cols.get(0);
if (col != null) { if (col != null) {
if (col.getJavaType() != JavaTypes.OBJECT) if (col.getJavaType() != JavaTypes.OBJECT)
return col.getJavaType(); return col.getJavaType();
@ -88,29 +92,67 @@ public class ValueMapDiscriminatorStrategy
@Override @Override
protected Class getClass(Object val, JDBCStore store) protected Class getClass(Object val, JDBCStore store)
throws ClassNotFoundException { throws ClassNotFoundException {
if(_vals == null) { if(_vals == null) {
_writeLock.lock();
try {
if(_vals == null) {
_vals = constructCache(disc);
}
} finally {
_writeLock.unlock();
}
}
String className = (val == null) ? null : val.toString();
_readLock.lock();
try {
Class<?> clz = _vals.get(className);
if (clz != null)
return clz;
} finally {
_readLock.unlock();
}
_writeLock.lock();
try {
Class<?> clz = _vals.get(className);
if (clz != null)
return clz;
//Rebuild the cache to check for updates
_vals = constructCache(disc);
//Try get again
clz = _vals.get(className);
if (clz != null)
return clz;
throw new ClassNotFoundException(_loc.get("unknown-discrim-value",
new Object[]{ className, disc.getClassMapping().getDescribedType().
getName(), new TreeSet<String>(_vals.keySet()) }).getMessage());
} finally {
_writeLock.unlock();
}
}
/**
* Build a class cache map from the discriminator
*/
private static Map<String, Class<?>> constructCache(Discriminator disc) {
//Build the cache map
ClassMapping cls = disc.getClassMapping(); ClassMapping cls = disc.getClassMapping();
ClassMapping[] subs = cls.getJoinablePCSubclassMappings(); ClassMapping[] subs = cls.getJoinablePCSubclassMappings();
Map map = new HashMap((int) ((subs.length + 1) * 1.33 + 1)); Map<String, Class<?>> map = new HashMap<String, Class<?>>((int) ((subs.length + 1) * 1.33 + 1));
mapDiscriminatorValue(cls, map); mapDiscriminatorValue(cls, map);
for (int i = 0; i < subs.length; i++) for (int i = 0; i < subs.length; i++)
mapDiscriminatorValue(subs[i], map); mapDiscriminatorValue(subs[i], map);
_vals = map; return map;
}
String str = (val == null) ? null : val.toString();
Class cls = (Class) _vals.get(str);
if (cls != null)
return cls;
throw new ClassNotFoundException(_loc.get("unknown-discrim-value",
new Object[]{ str, disc.getClassMapping().getDescribedType().
getName(), new TreeSet(_vals.keySet()) }).getMessage());
} }
/** /**
* Map the stringified version of the discriminator value of the given type. * Map the stringified version of the discriminator value of the given type.
*/ */
private static void mapDiscriminatorValue(ClassMapping cls, Map map) { private static void mapDiscriminatorValue(ClassMapping cls, Map<String, Class<?>> map) {
// possible that some types will never be persisted and therefore // possible that some types will never be persisted and therefore
// can have no discriminator value // can have no discriminator value
Object val = cls.getDiscriminator().getValue(); Object val = cls.getDiscriminator().getValue();
@ -118,7 +160,7 @@ public class ValueMapDiscriminatorStrategy
return; return;
String str = (val == Discriminator.NULL) ? null : val.toString(); String str = (val == Discriminator.NULL) ? null : val.toString();
Class exist = (Class) map.get(str); Class<?> exist = map.get(str);
if (exist != null) if (exist != null)
throw new MetaDataException(_loc.get("dup-discrim-value", throw new MetaDataException(_loc.get("dup-discrim-value",
str, exist, cls)); str, exist, cls));

View File

@ -126,7 +126,8 @@ public class MetaDataRepository implements PCRegistry.RegisterClassListener, Con
private Map<Class<?>, Class<?>> _metamodel = Collections.synchronizedMap(new HashMap<Class<?>, Class<?>>()); private Map<Class<?>, Class<?>> _metamodel = Collections.synchronizedMap(new HashMap<Class<?>, Class<?>>());
// map of classes to lists of their subclasses // map of classes to lists of their subclasses
private Map<Class<?>, List<Class<?>>> _subs = Collections.synchronizedMap(new HashMap<Class<?>, List<Class<?>>>()); private Map<Class<?>, Collection<Class<?>>> _subs =
Collections.synchronizedMap(new HashMap<Class<?>, Collection<Class<?>>>());
// xml mapping // xml mapping
protected final XMLMetaData[] EMPTY_XMLMETAS; protected final XMLMetaData[] EMPTY_XMLMETAS;
@ -1626,18 +1627,19 @@ public class MetaDataRepository implements PCRegistry.RegisterClassListener, Con
/** /**
* Updates our data structures with the latest registered classes. * Updates our data structures with the latest registered classes.
*
* This method is synchronized to make sure that all data structures are fully updated
* before other threads attempt to call this method
*/ */
Class<?>[] processRegisteredClasses(ClassLoader envLoader) { synchronized Class<?>[] processRegisteredClasses(ClassLoader envLoader) {
if (_registered.isEmpty()) if (_registered.isEmpty()) {
return EMPTY_CLASSES; return EMPTY_CLASSES;
}
// copy into new collection to avoid concurrent mod errors on reentrant // copy into new collection to avoid concurrent mod errors on reentrant
// registrations // registrations
Class<?>[] reg; Class<?>[] reg = _registered.toArray(new Class[_registered.size()]);
synchronized (_registered) {
reg = _registered.toArray(new Class[_registered.size()]);
_registered.clear(); _registered.clear();
}
Collection<String> pcNames = getPersistentTypeNames(false, envLoader); Collection<String> pcNames = getPersistentTypeNames(false, envLoader);
Collection<Class<?>> failed = null; Collection<Class<?>> failed = null;
@ -1840,7 +1842,7 @@ public class MetaDataRepository implements PCRegistry.RegisterClassListener, Con
/** /**
* Add the given value to the collection cached in the given map under the given key. * Add the given value to the collection cached in the given map under the given key.
*/ */
private void addToCollection(Map map, Class<?> key, Class<?> value, boolean inheritance) { private void addToCollection(Map<Class<?>, Collection<Class<?>>> map, Class<?> key, Class<?> value, boolean inheritance) {
if (_locking) { if (_locking) {
synchronized (map) { synchronized (map) {
addToCollectionInternal(map, key, value, inheritance); addToCollectionInternal(map, key, value, inheritance);
@ -1850,8 +1852,8 @@ public class MetaDataRepository implements PCRegistry.RegisterClassListener, Con
} }
} }
private void addToCollectionInternal(Map map, Class<?> key, Class<?> value, boolean inheritance) { private void addToCollectionInternal(Map<Class<?>, Collection<Class<?>>> map, Class<?> key, Class<?> value, boolean inheritance) {
Collection coll = (Collection) map.get(key); Collection<Class<?>> coll = map.get(key);
if (coll == null) { if (coll == null) {
if (inheritance) { if (inheritance) {
InheritanceComparator comp = new InheritanceComparator(); InheritanceComparator comp = new InheritanceComparator();