org.datanucleus.metadata.MetaDataManagerImpl Maven / Gradle / Ivy
Show all versions of datanucleus-core Show documentation
/**********************************************************************
Copyright (c) 2004 Andy Jefferson and others. All rights reserved.
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.
Contributors:
...
**********************************************************************/
package org.datanucleus.metadata;
import java.io.File;
import java.io.IOException;
import java.io.Serializable;
import java.net.URI;
import java.net.URL;
import java.security.AccessController;
import java.security.PrivilegedAction;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.Comparator;
import java.util.Enumeration;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.Properties;
import java.util.Set;
import java.util.TreeSet;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;
import org.datanucleus.ClassLoaderResolver;
import org.datanucleus.NucleusContext;
import org.datanucleus.PersistenceNucleusContext;
import org.datanucleus.PropertyNames;
import org.datanucleus.api.ApiAdapter;
import org.datanucleus.enhancer.EnhancementNucleusContextImpl;
import org.datanucleus.exceptions.ClassNotResolvedException;
import org.datanucleus.exceptions.NoPersistenceInformationException;
import org.datanucleus.exceptions.NucleusException;
import org.datanucleus.exceptions.NucleusUserException;
import org.datanucleus.metadata.annotations.AnnotationManager;
import org.datanucleus.metadata.annotations.AnnotationManagerImpl;
import org.datanucleus.util.ClassUtils;
import org.datanucleus.util.Localiser;
import org.datanucleus.util.NucleusLogger;
import org.datanucleus.util.StringUtils;
/**
* Manager of metadata information in DataNucleus having scope of an NucleusContext.
* Each PMF/EMF will effectively have a single MetaDataManager handling all XML/Annotations metadata.
*
* Metadata can be loaded into the MetaDataManager in two ways
*
* - Load when required. When the persistence process needs a class it will ask for metadata, and we can go and find its metadata from XML/annotations.
* - Load explicitly via API calls. This happens when handling persistence for a persistence-unit for example since we know what classes/mapping is involved.
* It is also the case with the enhancer where we know what classes to enhance so we load the metadata first
*
* The property allowMetaDataLoad is used to distinguish which of these is permitted.
* Similarly there are only certain methods that are available to load metadata during operation. The updateLock is used to lock
* access on these methods.
*
* Acts as a registry of metadata so that metadata files don't need to be parsed multiple times. MetaData is stored as a FileMetaData, which contains
* PackageMetaData, which contains ClassMetaData, and so on. This maps exactly to the users model of their metadata. The users access point is
* getMetaDataForClass() which will check the known classes without metadata, then check the existing registered metdata, then check the
* valid locations for metdata files. This way, the metadata is managed from this single point.
*
* Maintains a list of all classes that have been checked for MetaData and don't have any available. This avoids the needs to look up MetaData multiple times
* finding the same result. Currently this list is for all ClassMetaData objects keyed by the class name.
*
* Users can register interest in knowing when metadata for classes are loaded by registering a listener using the addListener method.
* This will then notify the listener when metadata for any class is initialised. This provides the opportunity to reject the metadata where particular features are
* not supported. For example a StoreManager could register a listener where it doesn't support datastore identity and throw an InvalidMetaDataException.
* This would then filter back out to the user for the operation they invoked
*
* MetaDataManager is intended to be thread-safe. All maps are ConcurrentHashMap to provide basic multithread usage.
* In addition all mutating methods make use of an update "lock" so that only one thread can update the metadata definition at any time.
*/
public abstract class MetaDataManagerImpl implements Serializable, MetaDataManager
{
private static final long serialVersionUID = 5223949693488111123L;
/** The NucleusContext that this metadata manager is operating in. */
protected final NucleusContext nucleusContext;
/** Manager for annotations. */
protected final AnnotationManager annotationManager;
/** Flag whether we should validate the (XML) metadata files when parsing. */
protected boolean validateXML = true;
/** Flag whether we should be (XML) namespace aware when parsing. */
protected boolean supportXMLNamespaces = true;
/** Flag defining if we allow metadata load currently. If false then we only allow access to currently loaded metadata. */
protected boolean allowMetaDataLoad = true;
/** Whether we allow XML metadata. */
protected boolean allowXML = true;
/** Whether we allow annotations metadata. */
protected boolean allowAnnotations = true;
/** Whether we allow ORM XML metadata. */
protected boolean allowORM = true;
/** Flag defining the default nullability for fields. */
protected boolean defaultNullable = true;
protected Lock updateLock = null;
/** Cache of class names that are known to not have MetaData/annotations. */
protected Collection classesWithoutPersistenceInfo = new HashSet<>();
/** Map of ClassMetaData, keyed by the class name. */
protected Map classMetaDataByClass = new ConcurrentHashMap<>();
protected Map usableClassMetaDataByClass = new ConcurrentHashMap<>();
/** Map of FileMetaData for the parsed files, keyed by the URL string. */
protected Map fileMetaDataByURLString = new ConcurrentHashMap<>();
/** Map of ClassMetaData, keyed by the JPA "entity name". */
protected Map classMetaDataByEntityName = new ConcurrentHashMap<>();
/** Map of ClassMetaData, keyed by the class discriminator name. */
protected Map classMetaDataByDiscriminatorName = new ConcurrentHashMap<>();
/** Cache subclass information as that is expensive to compute, keyed by class name */
protected Map> directSubclassesByClass = new ConcurrentHashMap<>();
/** Cache of names of concrete subclass for a class name. Used for fast lookups from ids. */
protected Map> concreteSubclassNamesByClassName = new ConcurrentHashMap<>();
/** Map of QueryMetaData, keyed by the (class name + query name). */
protected Map queryMetaDataByName = null;
/** Map of StoredProcQueryMetaData, keyed by the (class name + query name). */
protected Map storedProcQueryMetaDataByName = null;
/** Map of FetchPlanMetaData, keyed by the fetch plan name. */
protected Map fetchPlanMetaDataByName = null;
/** Map of SequenceMetaData, keyed by the package name and sequence name. */
protected Map sequenceMetaDataByPackageSequence = null;
/** Map of TableGeneratorMetaData, keyed by the package name and generator name. */
protected Map tableGeneratorMetaDataByPackageSequence = null;
/** Map of QueryResultMetaData keyed by the name. */
protected Map queryResultMetaDataByName = null;
/** Map of class metadata, keyed by the application-id object-id class name (not SingleField). */
protected Map> classMetaDataByAppIdClassName = new HashMap<>();
/** Listeners for metadata load. */
protected Set listeners = null;
/** Temporary list of class metadata loaded during the current call, for use with listeners. */
private List listenersLoadedMetaData = null;
/** Number of user metadata items. */
protected int userMetaDataNumber = 0;
protected Map discriminatorLookupByRootClassName = new ConcurrentHashMap();
private class DiscriminatorLookup
{
Map discrimValueByClass = new HashMap();
Map discrimClassByValue = new HashMap();
public void addValue(String className, String value)
{
this.discrimValueByClass.put(className, value);
this.discrimClassByValue.put(value, className);
}
public String getValueForClass(String className)
{
return discrimValueByClass.get(className);
}
public String getClassForValue(String value)
{
return discrimClassByValue.get(value);
}
public String toString()
{
return StringUtils.mapToString(discrimValueByClass);
}
}
/**
* Constructor, specifying the context used.
* @param ctx context that this metadata manager operates in
*/
public MetaDataManagerImpl(NucleusContext ctx)
{
nucleusContext = ctx;
updateLock = new ReentrantLock();
validateXML = nucleusContext.getConfiguration().getBooleanProperty(PropertyNames.PROPERTY_METADATA_XML_VALIDATE);
supportXMLNamespaces = nucleusContext.getConfiguration().getBooleanProperty(PropertyNames.PROPERTY_METADATA_XML_NAMESPACE_AWARE);
allowXML = nucleusContext.getConfiguration().getBooleanProperty(PropertyNames.PROPERTY_METADATA_ALLOW_XML);
allowAnnotations = nucleusContext.getConfiguration().getBooleanProperty(PropertyNames.PROPERTY_METADATA_ALLOW_ANNOTATIONS);
defaultNullable = nucleusContext.getConfiguration().getBooleanProperty(PropertyNames.PROPERTY_METADATA_DEFAULT_NULLABLE);
annotationManager = new AnnotationManagerImpl(this);
// Register all of the types managed by the TypeManager as known second-class types (no metadata).
Set supportedClasses = nucleusContext.getTypeManager().getSupportedSecondClassTypes();
for (String supportedClass : supportedClasses)
{
classesWithoutPersistenceInfo.add(supportedClass);
}
allowORM = nucleusContext.supportsORMMetaData();
if (allowORM)
{
Boolean configOrm = nucleusContext.getConfiguration().getBooleanObjectProperty(PropertyNames.PROPERTY_METADATA_SUPPORT_ORM);
if (configOrm != null && !configOrm.booleanValue())
{
// User has turned it off
allowORM = false;
}
}
}
/* (non-Javadoc)
* @see org.datanucleus.metadata.MetaDataManager#close()
*/
@Override
public void close()
{
classMetaDataByClass.clear();
classMetaDataByClass = null;
usableClassMetaDataByClass.clear();
usableClassMetaDataByClass = null;
fileMetaDataByURLString.clear();
fileMetaDataByURLString = null;
classesWithoutPersistenceInfo.clear();
classesWithoutPersistenceInfo = null;
directSubclassesByClass.clear();
directSubclassesByClass = null;
concreteSubclassNamesByClassName.clear();
concreteSubclassNamesByClassName = null;
if (classMetaDataByEntityName != null)
{
classMetaDataByEntityName.clear();
classMetaDataByEntityName = null;
}
if (classMetaDataByDiscriminatorName != null)
{
classMetaDataByDiscriminatorName.clear();
classMetaDataByDiscriminatorName = null;
}
if (queryMetaDataByName != null)
{
queryMetaDataByName.clear();
queryMetaDataByName = null;
}
if (storedProcQueryMetaDataByName != null)
{
storedProcQueryMetaDataByName.clear();
storedProcQueryMetaDataByName = null;
}
if (fetchPlanMetaDataByName != null)
{
fetchPlanMetaDataByName.clear();
fetchPlanMetaDataByName = null;
}
if (sequenceMetaDataByPackageSequence != null)
{
sequenceMetaDataByPackageSequence.clear();
sequenceMetaDataByPackageSequence = null;
}
if (tableGeneratorMetaDataByPackageSequence != null)
{
tableGeneratorMetaDataByPackageSequence.clear();
tableGeneratorMetaDataByPackageSequence = null;
}
if (queryResultMetaDataByName != null)
{
queryResultMetaDataByName.clear();
queryResultMetaDataByName = null;
}
if (classMetaDataByAppIdClassName != null)
{
classMetaDataByAppIdClassName.clear();
classMetaDataByAppIdClassName = null;
}
if (listeners != null)
{
listeners.clear();
listeners = null;
}
}
/* (non-Javadoc)
* @see org.datanucleus.metadata.MetaDataManager#registerListener(org.datanucleus.metadata.MetaDataListener)
*/
@Override
public void registerListener(MetaDataListener listener)
{
if (listeners == null)
{
listeners = new HashSet<>();
}
listeners.add(listener);
}
/* (non-Javadoc)
* @see org.datanucleus.metadata.MetaDataManager#deregisterListener(org.datanucleus.metadata.MetaDataListener)
*/
@Override
public void deregisterListener(MetaDataListener listener)
{
if (listeners == null)
{
return;
}
listeners.remove(listener);
if (listeners.isEmpty())
{
listeners = null;
}
}
/* (non-Javadoc)
* @see org.datanucleus.metadata.MetaDataManager#getEnhancedMethodNamePrefix()
*/
@Override
public String getEnhancedMethodNamePrefix()
{
return "dn";
}
public boolean isEnhancerField(String fieldName)
{
String prefix = "dn";
if (!fieldName.startsWith(prefix))
{
return false;
}
// TODO Take these from org.datanucleus.enhancer class at some point to avoid hardcoding
if (fieldName.equals("dnStateManager") || fieldName.equals("dnFlags") || fieldName.equals("dnDetachedState"))
{
return true;
}
// Static fields - commented out. Uncomment if we ever support persisting static fields
/*if (fieldName.equals("dnFieldFlags") ||
fieldName.equals("dnFieldNames") ||
fieldName.equals("dnFieldTypes") ||
fieldName.equals("dnPersistableSuperclass") ||
fieldName.equals("dnInheritedFieldCount"))
{
return true;
}*/
return false;
}
/* (non-Javadoc)
* @see org.datanucleus.metadata.MetaDataManager#setAllowMetaDataLoad(boolean)
*/
@Override
public void setAllowMetaDataLoad(boolean allow)
{
allowMetaDataLoad = allow;
}
/* (non-Javadoc)
* @see org.datanucleus.metadata.MetaDataManager#setAllowXML(boolean)
*/
@Override
public void setAllowXML(boolean allow)
{
this.allowXML = allow;
}
@Override
public void setDefaultNullable(boolean defaultNullable)
{
this.defaultNullable = defaultNullable;
}
/* (non-Javadoc)
* @see org.datanucleus.metadata.MetaDataManager#setAllowAnnotations(boolean)
*/
@Override
public void setAllowAnnotations(boolean allow)
{
this.allowAnnotations = allow;
}
/* (non-Javadoc)
* @see org.datanucleus.metadata.MetaDataManager#supportsORM()
*/
@Override
public boolean supportsORM()
{
return allowORM;
}
@Override
public boolean isDefaultNullable()
{
return defaultNullable;
}
/* (non-Javadoc)
* @see org.datanucleus.metadata.MetaDataManager#isEnhancing()
*/
@Override
public boolean isEnhancing()
{
return getNucleusContext() instanceof EnhancementNucleusContextImpl;
}
/* (non-Javadoc)
* @see org.datanucleus.metadata.MetaDataManager#setValidate(boolean)
*/
@Override
public void setValidate(boolean validate)
{
validateXML = validate;
}
/* (non-Javadoc)
* @see org.datanucleus.metadata.MetaDataManager#setXmlNamespaceAware(boolean)
*/
@Override
public void setXmlNamespaceAware(boolean aware)
{
supportXMLNamespaces = aware;
}
/* (non-Javadoc)
* @see org.datanucleus.metadata.MetaDataManager#getNucleusContext()
*/
@Override
public NucleusContext getNucleusContext()
{
return nucleusContext;
}
/* (non-Javadoc)
* @see org.datanucleus.metadata.MetaDataManager#getApiAdapter()
*/
@Override
public ApiAdapter getApiAdapter()
{
return nucleusContext.getApiAdapter();
}
/* (non-Javadoc)
* @see org.datanucleus.metadata.MetaDataManager#getAnnotationManager()
*/
@Override
public AnnotationManager getAnnotationManager()
{
return annotationManager;
}
/* (non-Javadoc)
* @see org.datanucleus.metadata.MetaDataManager#loadMetaDataFiles(java.lang.String[], java.lang.ClassLoader)
*/
@Override
public FileMetaData[] loadMetaDataFiles(String[] metadataFiles, ClassLoader loader)
{
if (!allowMetaDataLoad)
{
return null;
}
boolean originatingLoadCall = false;
if (listenersLoadedMetaData == null && listeners != null)
{
originatingLoadCall = true;
listenersLoadedMetaData = new ArrayList();
}
try
{
if (originatingLoadCall)
{
updateLock.lock();
}
if (NucleusLogger.METADATA.isDebugEnabled())
{
NucleusLogger.METADATA.debug(Localiser.msg("044005", StringUtils.objectArrayToString(metadataFiles)));
}
// Load MetaData files - will throw NucleusUserException if problems found
ClassLoaderResolver clr = nucleusContext.getClassLoaderResolver(loader);
Collection fileMetaData = loadFiles(metadataFiles, clr);
if (!fileMetaData.isEmpty())
{
// Populate/Initialise all loaded FileMetaData
initialiseFileMetaDataForUse(fileMetaData, clr);
}
if (NucleusLogger.METADATA.isDebugEnabled())
{
NucleusLogger.METADATA.debug(Localiser.msg("044010"));
}
if (originatingLoadCall)
{
processListenerLoadingCall();
}
return fileMetaData.toArray(new FileMetaData[fileMetaData.size()]);
}
finally
{
if (originatingLoadCall)
{
updateLock.unlock();
}
}
}
/* (non-Javadoc)
* @see org.datanucleus.metadata.MetaDataManager#loadClasses(java.lang.String[], java.lang.ClassLoader)
*/
@Override
public FileMetaData[] loadClasses(String[] classNames, ClassLoader loader)
{
if (!allowMetaDataLoad)
{
return null;
}
boolean originatingLoadCall = false;
if (listenersLoadedMetaData == null && listeners != null)
{
originatingLoadCall = true;
listenersLoadedMetaData = new ArrayList();
}
try
{
if (originatingLoadCall)
{
updateLock.lock();
}
if (NucleusLogger.METADATA.isDebugEnabled())
{
NucleusLogger.METADATA.debug(Localiser.msg("044006", StringUtils.objectArrayToString(classNames)));
}
// Load classes
ClassLoaderResolver clr = nucleusContext.getClassLoaderResolver(loader);
Collection fileMetaData = new ArrayList<>();
Set exceptions = new HashSet<>();
for (String className : classNames)
{
try
{
Class cls = clr.classForName(className);
// Check for MetaData for this class (take precedence over annotations if they exist)
AbstractClassMetaData cmd = classMetaDataByClass.get(className);
if (cmd == null)
{
// No MetaData so try annotations
FileMetaData filemd = loadAnnotationsForClass(cls, clr, true, false);
if (filemd != null)
{
// Store file against an annotations specific "URL"
registerFile("annotations:" + className, filemd, clr);
fileMetaData.add(filemd);
}
else
{
cmd = getMetaDataForClass(cls, clr);
if (cmd == null)
{
// Class has no metadata or annotations so warn the user
NucleusLogger.METADATA.debug(Localiser.msg("044017", className));
}
else
{
fileMetaData.add(cmd.getPackageMetaData().getFileMetaData());
}
}
}
else
{
fileMetaData.add(cmd.getPackageMetaData().getFileMetaData());
// We have MetaData, and any annotations will be merged in during the populate process
}
}
catch (ClassNotResolvedException e)
{
// log and ignore this exception
NucleusLogger.METADATA.error(StringUtils.getStringFromStackTrace(e));
}
catch (Exception e)
{
exceptions.add(e);
}
}
if (!exceptions.isEmpty())
{
// Exceptions while loading annotations
throw new NucleusUserException(Localiser.msg("044016"), exceptions.toArray(new Throwable[exceptions.size()]),null);
}
if (!fileMetaData.isEmpty())
{
// Populate/Initialise all loaded FileMetaData
initialiseFileMetaDataForUse(fileMetaData, clr);
}
if (NucleusLogger.METADATA.isDebugEnabled())
{
NucleusLogger.METADATA.debug(Localiser.msg("044010"));
}
if (originatingLoadCall)
{
processListenerLoadingCall();
}
return fileMetaData.toArray(new FileMetaData[fileMetaData.size()]);
}
finally
{
if (originatingLoadCall)
{
updateLock.unlock();
}
}
}
/* (non-Javadoc)
* @see org.datanucleus.metadata.MetaDataManager#loadJar(java.lang.String, java.lang.ClassLoader)
*/
@Override
public FileMetaData[] loadJar(String jarFileName, ClassLoader loader)
{
if (!allowMetaDataLoad)
{
return null;
}
boolean originatingLoadCall = false;
if (listenersLoadedMetaData == null && listeners != null)
{
originatingLoadCall = true;
listenersLoadedMetaData = new ArrayList();
}
try
{
if (originatingLoadCall)
{
updateLock.lock();
}
if (NucleusLogger.METADATA.isDebugEnabled())
{
NucleusLogger.METADATA.debug(Localiser.msg("044009", jarFileName));
}
ClassLoaderResolver clr = nucleusContext.getClassLoaderResolver(loader);
List fileMetaData = new ArrayList<>();
// Generate list of package.jdo and classes present in the jar
Set mappingFileNames = new HashSet<>();
if (allowXML)
{
String[] packageJdoFiles = ClassUtils.getPackageJdoFilesForJarFile(jarFileName);
if (packageJdoFiles != null)
{
for (String packageJdoFile : packageJdoFiles)
{
mappingFileNames.add(packageJdoFile);
}
}
}
Set classNames = new HashSet<>();
if (allowAnnotations)
{
String[] jarClassNames = ClassUtils.getClassNamesForJarFile(jarFileName);
if (jarClassNames != null)
{
for (String jarClassName : jarClassNames)
{
classNames.add(jarClassName);
}
}
}
Set exceptions = new HashSet<>();
if (allowXML && !mappingFileNames.isEmpty())
{
// Load XML metadata
for (String mappingFileName : mappingFileNames)
{
try
{
Enumeration files = clr.getResources(mappingFileName, Thread.currentThread().getContextClassLoader());
while (files.hasMoreElements())
{
URL url = (URL)files.nextElement();
if (url != null && fileMetaDataByURLString.get(url.toString()) == null)
{
FileMetaData filemd = parseXmlFile(url);
if (filemd != null)
{
// Register the file
registerFile(url.toString(), filemd, clr);
fileMetaData.add(filemd);
}
}
}
}
catch (InvalidMetaDataException imde)
{
// Error in the metadata for this file
NucleusLogger.METADATA.error(StringUtils.getStringFromStackTrace(imde));
exceptions.add(imde);
}
catch (IOException ioe)
{
NucleusLogger.METADATA.error(Localiser.msg("044027", jarFileName, mappingFileName, ioe.getMessage()), ioe);
}
}
}
if (allowAnnotations && !classNames.isEmpty())
{
// Load annotation metadata for all classes
for (String className : classNames)
{
// Check for MetaData for this class (take precedence over annotations if they exist)
AbstractClassMetaData cmd = classMetaDataByClass.get(className);
if (cmd == null)
{
// No MetaData so try annotations
try
{
Class cls = clr.classForName(className);
FileMetaData filemd = loadAnnotationsForClass(cls, clr, true, false);
if (filemd != null)
{
fileMetaData.add(filemd);
}
}
catch (ClassNotResolvedException e)
{
// log and ignore this exception
NucleusLogger.METADATA.error(StringUtils.getStringFromStackTrace(e));
}
catch (Throwable e)
{
exceptions.add(e);
}
}
else
{
// We have MetaData, and any annotations will be merged in during the populate process
}
}
}
if (!exceptions.isEmpty())
{
throw new NucleusUserException(Localiser.msg("044024", jarFileName), exceptions.toArray(new Throwable[exceptions.size()]));
}
if (!fileMetaData.isEmpty())
{
// Populate/Initialise all loaded FileMetaData
initialiseFileMetaDataForUse(fileMetaData, clr);
}
if (NucleusLogger.METADATA.isDebugEnabled())
{
NucleusLogger.METADATA.debug(Localiser.msg("044010"));
}
if (originatingLoadCall)
{
processListenerLoadingCall();
}
return fileMetaData.toArray(new FileMetaData[fileMetaData.size()]);
}
finally
{
if (originatingLoadCall)
{
updateLock.unlock();
}
}
}
/* (non-Javadoc)
* @see org.datanucleus.metadata.MetaDataManager#loadPersistenceUnit(org.datanucleus.metadata.PersistenceUnitMetaData, java.lang.ClassLoader)
*/
@Override
public FileMetaData[] loadPersistenceUnit(PersistenceUnitMetaData pumd, ClassLoader loader)
{
if (!allowMetaDataLoad)
{
return null;
}
boolean originatingLoadCall = false;
if (listenersLoadedMetaData == null && listeners != null)
{
originatingLoadCall = true;
listenersLoadedMetaData = new ArrayList();
}
try
{
if (originatingLoadCall)
{
updateLock.lock();
}
if (NucleusLogger.METADATA.isDebugEnabled())
{
NucleusLogger.METADATA.debug(Localiser.msg("044007", pumd.getName()));
}
Properties puProps = pumd.getProperties();
if (puProps != null)
{
// Apply any properties from this persistence unit to the metadata manager
if (puProps.containsKey(PropertyNames.PROPERTY_METADATA_XML_VALIDATE))
{
Boolean val = Boolean.valueOf((String) puProps.get(PropertyNames.PROPERTY_METADATA_XML_VALIDATE));
if (val != null)
{
validateXML = val;
}
}
if (puProps.containsKey(PropertyNames.PROPERTY_METADATA_XML_NAMESPACE_AWARE))
{
Boolean val = Boolean.valueOf((String) puProps.get(PropertyNames.PROPERTY_METADATA_XML_NAMESPACE_AWARE));
if (val != null)
{
supportXMLNamespaces = val;
}
}
}
ClassLoaderResolver clr = nucleusContext.getClassLoaderResolver(loader);
Set exceptions = new HashSet<>();
List fileMetaData = new ArrayList<>();
// Generate list of XML files
Set mappingFileNames = new HashSet<>();
if (allowXML)
{
String defaultMappingFile = nucleusContext.getApiAdapter().getDefaultMappingFileLocation();
if (defaultMappingFile != null)
{
mappingFileNames.add(defaultMappingFile);
}
if (pumd.getMappingFiles() != null)
{
//
mappingFileNames.addAll(pumd.getMappingFiles());
}
if (nucleusContext.getApiName().equalsIgnoreCase("JDO")) // When in JDO mode grab any package.jdo TODO Abstract this into api-jdo and ApiAdapter
{
//
Set jarFileNames = pumd.getJarFiles();
if (jarFileNames != null)
{
Iterator iter = jarFileNames.iterator();
while (iter.hasNext())
{
Object jarFile = iter.next();
if (jarFile instanceof String)
{
String[] packageJdoFiles = ClassUtils.getPackageJdoFilesForJarFile((String)jarFile);
if (packageJdoFiles != null)
{
for (String packageJdoFile : packageJdoFiles)
{
mappingFileNames.add(packageJdoFile);
}
}
}
else if (jarFile instanceof URL)
{
String[] packageJdoFiles = ClassUtils.getPackageJdoFilesForJarFile((URL)jarFile);
if (packageJdoFiles != null)
{
for (String packageJdoFile : packageJdoFiles)
{
mappingFileNames.add(packageJdoFile);
}
}
}
else if (jarFile instanceof URI)
{
String[] packageJdoFiles = ClassUtils.getPackageJdoFilesForJarFile((URI)jarFile);
if (packageJdoFiles != null)
{
for (String packageJdoFile : packageJdoFiles)
{
mappingFileNames.add(packageJdoFile);
}
}
}
}
}
}
}
// Generate list of (possibly annotated) class names
Set classNames = new HashSet<>();
if (allowAnnotations)
{
if (pumd.getClassNames() != null)
{
classNames.addAll(pumd.getClassNames());
}
// TODO Process pumd.getConverters
if (getNucleusContext() instanceof PersistenceNucleusContext) // TODO Why not when enhancing? document it
{
Set jarFileNames = pumd.getJarFiles();
if (jarFileNames != null)
{
Iterator iter = jarFileNames.iterator();
while (iter.hasNext())
{
Object jarFile = iter.next();
if (jarFile instanceof String)
{
String[] jarClassNames = ClassUtils.getClassNamesForJarFile((String)jarFile);
if (jarClassNames != null)
{
for (String jarClassName : jarClassNames)
{
classNames.add(jarClassName);
}
}
}
else if (jarFile instanceof URL)
{
String[] jarClassNames = ClassUtils.getClassNamesForJarFile((URL)jarFile);
if (jarClassNames != null)
{
for (String jarClassName : jarClassNames)
{
classNames.add(jarClassName);
}
}
}
else if (jarFile instanceof URI)
{
String[] jarClassNames = ClassUtils.getClassNamesForJarFile((URI)jarFile);
if (jarClassNames != null)
{
for (String jarClassName : jarClassNames)
{
classNames.add(jarClassName);
}
}
}
}
}
}
if (!pumd.getExcludeUnlistedClasses())
{
MetaDataScanner scanner = getScanner(clr);
if (scanner != null)
{
Set scannedClassNames = scanner.scanForPersistableClasses(pumd);
if (scannedClassNames != null)
{
classNames.addAll(scannedClassNames);
}
}
else
{
// Classpath scan for other classes
try
{
if (pumd.getRootURI() != null && pumd.getRootURI().getScheme().equals("file"))
{
// File-based root so load all classes under the root URL of the persistence-unit
File rootDir = new File(pumd.getRootURI());
String[] scannedClassNames = ClassUtils.getClassNamesForDirectoryAndBelow(rootDir);
if (scannedClassNames != null)
{
for (String scannedClassName : scannedClassNames)
{
NucleusLogger.METADATA.debug(Localiser.msg("044026", scannedClassName, pumd.getName()));
classNames.add(scannedClassName);
}
}
}
}
catch (IllegalArgumentException iae)
{
NucleusLogger.METADATA.debug("Ignoring scan of classes for this persistence-unit since the URI root is " + pumd.getRootURI() + " and is not hierarchical");
// Ignore the scan for classes
}
}
}
}
if (allowXML && !mappingFileNames.isEmpty())
{
// Load XML metadata for all specifications
for (String mappingFileName : mappingFileNames)
{
try
{
Enumeration files = clr.getResources(mappingFileName, Thread.currentThread().getContextClassLoader());
if (!files.hasMoreElements())
{
NucleusLogger.METADATA.debug("Not found any metadata mapping files for resource name " + mappingFileName + " in CLASSPATH");
}
else
{
while (files.hasMoreElements())
{
URL url = (URL)files.nextElement();
if (url != null && fileMetaDataByURLString.get(url.toString()) == null)
{
FileMetaData filemd = parseXmlFile(url);
if (filemd != null)
{
// Register the file
registerFile(url.toString(), filemd, clr);
fileMetaData.add(filemd);
}
}
}
}
}
catch (InvalidMetaDataException imde)
{
// Error in the metadata for this file
NucleusLogger.METADATA.error(StringUtils.getStringFromStackTrace(imde));
exceptions.add(imde);
}
catch (IOException ioe)
{
NucleusLogger.METADATA.error(Localiser.msg("044027", pumd.getName(), mappingFileName, ioe.getMessage()), ioe);
}
}
}
if (allowAnnotations && !classNames.isEmpty())
{
// Load annotation metadata for all classes
for (String className : classNames)
{
// Check for MetaData for this class (take precedence over annotations if they exist)
AbstractClassMetaData cmd = classMetaDataByClass.get(className);
if (cmd == null)
{
// No MetaData so try annotations
try
{
Class cls = clr.classForName(className);
FileMetaData filemd = loadAnnotationsForClass(cls, clr, true, false);
if (filemd != null)
{
fileMetaData.add(filemd);
}
else
{
NucleusLogger.METADATA.debug("Class " + className + " was specified in persistence-unit (maybe by not putting exclude-unlisted-classes) " +
pumd.getName() + " but not annotated, so ignoring");
}
}
catch (ClassNotResolvedException e)
{
// log and ignore this exception
NucleusLogger.METADATA.error(StringUtils.getStringFromStackTrace(e));
}
catch (Throwable e)
{
exceptions.add(e);
}
}
else
{
// We have MetaData, and any annotations will be merged in during the populate process
}
}
}
if (!exceptions.isEmpty())
{
throw new NucleusUserException(Localiser.msg("044023", pumd.getName()), exceptions.toArray(new Throwable[exceptions.size()]));
}
if (!fileMetaData.isEmpty())
{
// Populate/Initialise all loaded FileMetaData
initialiseFileMetaDataForUse(fileMetaData, clr);
}
// Some other (inner) classes may have been brought in the populate of the above so check
// TODO Really need a Set of unpopulated/uninitialised metadata and continue til all done
for (AbstractClassMetaData cmd : classMetaDataByClass.values())
{
boolean populated = cmd.isPopulated();
if (!cmd.isPopulated())
{
populated = populateAbstractClassMetaData(cmd, clr, loader);
}
if (populated && !cmd.isInitialised())
{
initialiseAbstractClassMetaData(cmd, clr);
}
}
if (NucleusLogger.METADATA.isDebugEnabled())
{
NucleusLogger.METADATA.debug(Localiser.msg("044010"));
}
if (originatingLoadCall)
{
processListenerLoadingCall();
}
return fileMetaData.toArray(new FileMetaData[fileMetaData.size()]);
}
finally
{
if (originatingLoadCall)
{
updateLock.unlock();
}
}
}
/* (non-Javadoc)
* @see org.datanucleus.metadata.MetaDataManager#loadUserMetaData(org.datanucleus.metadata.FileMetaData, java.lang.ClassLoader)
*/
@Override
public void loadUserMetaData(FileMetaData fileMetaData, ClassLoader loader)
{
if (fileMetaData == null)
{
return;
}
if (!allowMetaDataLoad)
{
return;
}
boolean originatingLoadCall = false;
if (listenersLoadedMetaData == null && listeners != null)
{
originatingLoadCall = true;
listenersLoadedMetaData = new ArrayList();
}
try
{
if (originatingLoadCall)
{
updateLock.lock();
}
if (NucleusLogger.METADATA.isDebugEnabled())
{
NucleusLogger.METADATA.debug(Localiser.msg("044008"));
}
ClassLoaderResolver clr = nucleusContext.getClassLoaderResolver(loader);
fileMetaData.setFilename("User_Metadata_" + userMetaDataNumber);
userMetaDataNumber++;
registerFile(fileMetaData.getFilename(), fileMetaData, clr);
Collection filemds = new ArrayList<>();
filemds.add(fileMetaData);
initialiseFileMetaDataForUse(filemds, clr);
if (NucleusLogger.METADATA.isDebugEnabled())
{
NucleusLogger.METADATA.debug(Localiser.msg("044010"));
}
if (originatingLoadCall)
{
processListenerLoadingCall();
}
}
finally
{
if (originatingLoadCall)
{
updateLock.unlock();
}
}
}
/**
* Load the metadata for the specified class (if available).
* @param c The class
* @param clr ClassLoader resolver
* @return The metadata for this class (if found)
*/
protected abstract AbstractClassMetaData loadMetaDataForClass(Class c, ClassLoaderResolver clr);
/* (non-Javadoc)
* @see org.datanucleus.metadata.MetaDataManager#unloadMetaDataForClass(java.lang.String)
*/
@Override
public void unloadMetaDataForClass(String className)
{
try
{
updateLock.lock();
// Remove any reference to the AbstractClassMetaData
usableClassMetaDataByClass.remove(className);
AbstractClassMetaData cmd = classMetaDataByClass.remove(className);
Iterator> iter = classMetaDataByDiscriminatorName.entrySet().iterator();
while (iter.hasNext())
{
Map.Entry entry = iter.next();
if (entry.getValue() == cmd)
{
iter.remove();
}
}
iter = classMetaDataByEntityName.entrySet().iterator();
while (iter.hasNext())
{
Map.Entry entry = iter.next();
if (entry.getValue() == cmd)
{
iter.remove();
}
}
Collection> classMetaDataByAppIdValues = classMetaDataByAppIdClassName.values();
Iterator> classMetaDataByAppIdValuesIter = classMetaDataByAppIdValues.iterator();
while (classMetaDataByAppIdValuesIter.hasNext())
{
Collection collCmds = classMetaDataByAppIdValuesIter.next();
if (!collCmds.isEmpty())
{
collCmds.remove(cmd);
}
}
// TODO Remove this class from subclass lookups of other classes
concreteSubclassNamesByClassName.remove(className);
directSubclassesByClass.remove(className);
discriminatorLookupByRootClassName.remove(className);
classesWithoutPersistenceInfo.remove(className);
}
finally
{
updateLock.unlock();
}
}
/**
* Accessor for any scanner for metadata classes (optional).
* Looks for the persistence property "datanucleus.metadata.scanner" (if defined)
* @param clr The classloader resolver
* @return scanner instance or null if it doesn't exist or cannot be instantiated
*/
protected MetaDataScanner getScanner(ClassLoaderResolver clr)
{
Object so = nucleusContext.getConfiguration().getProperty(PropertyNames.PROPERTY_METADATA_SCANNER);
if (so == null)
{
return null;
}
if (so instanceof MetaDataScanner)
{
return (MetaDataScanner)so;
}
else if (so instanceof String)
{
try
{
return (MetaDataScanner) clr.classForName((String)so).getDeclaredConstructor().newInstance();
}
catch (Throwable t)
{
throw new NucleusUserException(Localiser.msg("044012", so), t);
}
}
else
{
if (NucleusLogger.METADATA.isDebugEnabled())
{
NucleusLogger.METADATA.debug(Localiser.msg("044011", so));
}
return null;
}
}
/**
* Method to initialise the provided FileMetaData, ready for use.
* @param fileMetaData Collection of FileMetaData
* @param clr ClassLoader resolver
* @throws NucleusUserException thrown if an error occurs during the populate/initialise of the supplied metadata.
*/
protected void initialiseFileMetaDataForUse(Collection fileMetaData, ClassLoaderResolver clr)
{
Set exceptions = new HashSet<>();
// a). Populate MetaData
if (NucleusLogger.METADATA.isDebugEnabled())
{
NucleusLogger.METADATA.debug(Localiser.msg("044018"));
}
for (FileMetaData filemd : fileMetaData)
{
if (!filemd.isInitialised())
{
populateFileMetaData(filemd, clr, null);
}
}
// b). Initialise MetaData
if (NucleusLogger.METADATA.isDebugEnabled())
{
NucleusLogger.METADATA.debug(Localiser.msg("044019"));
}
for (FileMetaData filemd : fileMetaData)
{
if (!filemd.isInitialised())
{
try
{
initialiseFileMetaData(filemd, clr, null);
}
catch (Exception e)
{
NucleusLogger.METADATA.error(StringUtils.getStringFromStackTrace(e));
exceptions.add(e);
}
}
}
if (!exceptions.isEmpty())
{
throw new NucleusUserException(Localiser.msg("044020"), exceptions.toArray(new Throwable[exceptions.size()]));
}
}
/* (non-Javadoc)
* @see org.datanucleus.metadata.MetaDataManager#loadFiles(java.lang.String[], org.datanucleus.ClassLoaderResolver)
*/
@Override
public Collection loadFiles(String[] metadataFiles, ClassLoaderResolver clr)
{
List fileMetaData = new ArrayList<>();
Set exceptions = new HashSet<>();
if (allowXML)
{
for (String metadataFile : metadataFiles)
{
try
{
URL fileURL = null;
try
{
// Try as file
File file = new File(metadataFile);
fileURL = file.toURI().toURL();
if (!file.exists())
{
// Try as CLASSPATH resource
fileURL = clr.getResource(metadataFile, null);
}
}
catch (Exception mue)
{
// Try as CLASSPATH resource
fileURL = clr.getResource(metadataFile, null);
}
if (fileURL == null)
{
// User provided a filename which doesn't exist
NucleusLogger.METADATA.warn("Metadata file " + metadataFile + " not found in CLASSPATH");
continue;
}
FileMetaData filemd = fileMetaDataByURLString.get(fileURL.toString());
if (filemd == null)
{
// Valid metadata, and not already loaded
filemd = parseXmlFile(fileURL);
if (filemd != null)
{
registerFile(fileURL.toString(), filemd, clr);
fileMetaData.add(filemd);
}
else
{
throw new NucleusUserException(Localiser.msg("044015", metadataFile));
}
}
else
{
fileMetaData.add(filemd);
}
}
catch (Exception e)
{
NucleusLogger.METADATA.error(StringUtils.getStringFromStackTrace(e));
exceptions.add(e);
}
}
}
if (!exceptions.isEmpty())
{
// Exceptions while loading MetaData
throw new NucleusUserException(Localiser.msg("044016"), exceptions.toArray(new Throwable[exceptions.size()]), null);
}
return fileMetaData;
}
/* (non-Javadoc)
* @see org.datanucleus.metadata.MetaDataManager#isClassPersistable(java.lang.String)
*/
@Override
public boolean isClassPersistable(String className)
{
AbstractClassMetaData acmd = readMetaDataForClass(className);
if (acmd == null)
{
return false;
}
return acmd.getPersistenceModifier() == ClassPersistenceModifier.PERSISTENCE_CAPABLE;
}
/* (non-Javadoc)
* @see org.datanucleus.metadata.MetaDataManager#getFileMetaData()
*/
@Override
public FileMetaData[] getFileMetaData()
{
Collection filemds = fileMetaDataByURLString.values();
return filemds.toArray(new FileMetaData[filemds.size()]);
}
/* (non-Javadoc)
* @see org.datanucleus.metadata.MetaDataManager#getClassesWithMetaData()
*/
@Override
public Collection getClassesWithMetaData()
{
return Collections.unmodifiableCollection(classMetaDataByClass.keySet());
}
/* (non-Javadoc)
* @see org.datanucleus.metadata.MetaDataManager#hasMetaDataForClass(java.lang.String)
*/
@Override
public boolean hasMetaDataForClass(String className)
{
if (className == null)
{
return false;
}
// Check if this class has no MetaData before instantiating its class
if (isClassWithoutPersistenceInfo(className))
{
return false;
}
return classMetaDataByClass.get(className) != null;
}
/**
* Accessor for whether a class doesn't have MetaData or annotations.
* @param className Name of the class
* @return Whether it has no metadata and annotations
*/
protected boolean isClassWithoutPersistenceInfo(String className)
{
if (className == null)
{
return true;
}
// Standard Java classes have no MetaData
if (className.startsWith("java.") || className.startsWith("javax."))
{
return true;
}
// Use the cache to determine if it has metadata
return classesWithoutPersistenceInfo.contains(className);
}
/* (non-Javadoc)
* @see org.datanucleus.metadata.MetaDataManager#getClassMetaDataWithApplicationId(java.lang.String)
*/
@Override
public Collection getClassMetaDataWithApplicationId(String objectIdClassName)
{
return classMetaDataByAppIdClassName.get(objectIdClassName);
}
/* (non-Javadoc)
* @see org.datanucleus.metadata.MetaDataManager#getMetaDataForClass(java.lang.String, org.datanucleus.ClassLoaderResolver)
*/
@Override
public AbstractClassMetaData getMetaDataForClass(String className, ClassLoaderResolver clr)
{
if (className == null)
{
return null;
}
AbstractClassMetaData cmd = usableClassMetaDataByClass.get(className);
if (cmd != null)
{
return cmd;
}
// Check if this class has no MetaData/annotations before instantiating its class
if (isClassWithoutPersistenceInfo(className))
{
return null;
}
synchronized(this)
{
// Check if we have the MetaData already
cmd = classMetaDataByClass.get(className);
if (cmd != null && cmd.isPopulated() && cmd.isInitialised() && cmd instanceof ClassMetaData)
{
// We explicitly don't return metadata for persistent interfaces here since they should return the impl CMD
return cmd;
}
// Resolve the class
Class> c = null;
try
{
c = (clr == null) ? Class.forName(className) : clr.classForName(className, null, false);
}
catch (ClassNotFoundException | ClassNotResolvedException cne)
{
}
if (c == null)
{
if (cmd != null && cmd.isPopulated() && cmd.isInitialised())
{
// Return any previously loaded metadata
return cmd;
}
return null;
}
cmd = getMetaDataForClass(c, clr);
if (cmd != null)
{
usableClassMetaDataByClass.put(className, cmd);
}
return cmd;
}
}
/** Temporary list of the FileMetaData objects utilised in this call for metadata. */
protected ArrayList utilisedFileMetaData = new ArrayList<>();
/* (non-Javadoc)
* @see org.datanucleus.metadata.MetaDataManager#getMetaDataForClass(java.lang.Class, org.datanucleus.ClassLoaderResolver)
*/
@Override
public AbstractClassMetaData getMetaDataForClass(Class c, ClassLoaderResolver clr)
{
if (c == null)
{
return null;
}
AbstractClassMetaData cmd = usableClassMetaDataByClass.get(c.getName());
if (cmd != null)
{
return cmd;
}
// Check if this class has no MetaData/annotations before instantiating its class
if (isClassWithoutPersistenceInfo(c.getName()))
{
return null;
}
synchronized(this)
{
boolean originatingLoadCall = false;
if (listenersLoadedMetaData == null && listeners != null)
{
originatingLoadCall = true;
listenersLoadedMetaData = new ArrayList();
}
// Find the metadata if present, allowing it to be loaded
cmd = getMetaDataForClassInternal(c, clr);
if (c.isInterface())
{
// "persistent-interface" - check if it has class built at runtime and return the MetaData for it
cmd = getClassMetaDataForImplementationOfPersistentInterface(c.getName());
}
if (cmd != null)
{
// Make sure that anything returned is initialised
boolean populated = populateAbstractClassMetaData(cmd, clr, c.getClassLoader());
if (populated)
{
initialiseAbstractClassMetaData(cmd, clr);
}
// Make sure all FileMetaData that were subsequently loaded as a result of this call are all initialised before return
if (!utilisedFileMetaData.isEmpty())
{
// Pass 1 - initialise anything loaded during the initialise of the requested class
List utilisedFileMetaData1 = List.copyOf(utilisedFileMetaData);
utilisedFileMetaData.clear();
for (FileMetaData filemd : utilisedFileMetaData1)
{
initialiseFileMetaData(filemd, clr,c.getClassLoader());
}
if (!utilisedFileMetaData.isEmpty())
{
// Pass 2 - initialise anything loaded during the initialise of pass 1
List utilisedFileMetaData2 = List.copyOf(utilisedFileMetaData);
utilisedFileMetaData.clear();
for (FileMetaData filemd : utilisedFileMetaData2)
{
initialiseFileMetaData(filemd, clr,c.getClassLoader());
}
}
}
}
else
{
if (!c.isInterface())
{
classesWithoutPersistenceInfo.add(c.getName());
}
}
utilisedFileMetaData.clear();
if (originatingLoadCall)
{
processListenerLoadingCall();
}
if (cmd != null)
{
usableClassMetaDataByClass.put(c.getName(), cmd);
}
return cmd;
}
}
protected void processListenerLoadingCall()
{
if (!listenersLoadedMetaData.isEmpty() && listeners != null)
{
// Notify any listeners of the metadata loaded during this call
Iterator loadedIter = new ArrayList<>(listenersLoadedMetaData).iterator();
while (loadedIter.hasNext())
{
AbstractClassMetaData acmd = loadedIter.next();
for (MetaDataListener listener : listeners)
{
listener.loaded(acmd);
}
}
}
listenersLoadedMetaData = null;
}
/* (non-Javadoc)
* @see org.datanucleus.metadata.MetaDataManager#getMetaDataForEntityName(java.lang.String)
*/
@Override
public AbstractClassMetaData getMetaDataForEntityName(String entityName)
{
return classMetaDataByEntityName.get(entityName);
}
/* (non-Javadoc)
* @see org.datanucleus.metadata.MetaDataManager#getMetaDataForDiscriminator(java.lang.String)
*/
@Override
public AbstractClassMetaData getMetaDataForDiscriminator(String discriminator)
{
return classMetaDataByDiscriminatorName.get(discriminator);
}
/* (non-Javadoc)
* @see org.datanucleus.metadata.MetaDataManager#readMetaDataForClass(java.lang.String)
*/
@Override
public AbstractClassMetaData readMetaDataForClass(String className)
{
return classMetaDataByClass.get(className);
}
/* (non-Javadoc)
* @see org.datanucleus.metadata.MetaDataManager#readMetaDataForMember(java.lang.String, java.lang.String)
*/
@Override
public AbstractMemberMetaData readMetaDataForMember(String className, String memberName)
{
AbstractClassMetaData cmd = readMetaDataForClass(className);
return cmd != null ? cmd.getMetaDataForMember(memberName) : null;
}
/* (non-Javadoc)
* @see org.datanucleus.metadata.MetaDataManager#getMetaDataForClassInternal(java.lang.Class, org.datanucleus.ClassLoaderResolver)
*/
@Override
public AbstractClassMetaData getMetaDataForClassInternal(Class c, ClassLoaderResolver clr)
{
if (c.isArray())
{
// Only particular classes can have metadata
return null;
}
// If we know that this class/interface has no MetaData/annotations don't bother searching
if (isClassWithoutPersistenceInfo(c.getName()))
{
return null;
}
// Check if we have the MetaData loaded
AbstractClassMetaData cmd = classMetaDataByClass.get(c.getName());
if (cmd != null)
{
return cmd;
}
// Try to load metadata for this class
return loadMetaDataForClass(c, clr);
}
/**
* Internal method called when we want to register the metadata for a class/interface.
* @param fullClassName Name of the class
* @param cmd The metadata
*/
protected void registerMetaDataForClass(String fullClassName, AbstractClassMetaData cmd)
{
classMetaDataByClass.put(fullClassName, cmd);
}
/* (non-Javadoc)
* @see org.datanucleus.metadata.MetaDataManager#getClassesImplementingInterface(java.lang.String, org.datanucleus.ClassLoaderResolver)
*/
@Override
public String[] getClassesImplementingInterface(String interfaceName, ClassLoaderResolver clr)
{
Collection classes = new HashSet<>();
Class> intfClass = clr.classForName(interfaceName);
Collection generatedClassNames = new HashSet<>();
// Loop through all known classes and find the implementations
Collection cmds = classMetaDataByClass.values();
boolean isPersistentInterface = false;
for (AbstractClassMetaData acmd : cmds)
{
Class> implClass = null;
try
{
implClass = clr.classForName(acmd.getFullClassName());
}
catch (ClassNotResolvedException cnre)
{
// Implementation class not yet generated
}
if (implClass != null)
{
if (acmd instanceof ClassMetaData)
{
// Make sure that we are initialised since implementsMetaData wont be set
initialiseAbstractClassMetaData(acmd, clr);
if (intfClass.isAssignableFrom(implClass))
{
if (!((ClassMetaData)acmd).isAbstract())
{
classes.add(implClass);
}
}
}
else if (acmd instanceof InterfaceMetaData)
{
if (intfClass.isAssignableFrom(implClass))
{
isPersistentInterface = true;
}
}
}
else
{
if (isPersistentInterfaceImplementation(interfaceName, acmd.getFullClassName()))
{
isPersistentInterface = true;
generatedClassNames.add(acmd.getFullClassName());
}
}
}
if (isPersistentInterface && nucleusContext instanceof PersistenceNucleusContext && ((PersistenceNucleusContext)nucleusContext).getImplementationCreator() != null)
{
// JDO "persistent interfaces" - deliberately kept separate from normal persistence since it is largely undocumented and best left alone
// TODO this is very time consuming. got to do some cache
classes.add(((PersistenceNucleusContext)nucleusContext).getImplementationCreator().newInstance(intfClass, clr).getClass());
int numClasses = classes.size() + generatedClassNames.size();
String[] classNames = new String[numClasses];
int i = 0;
for (Class> cls : classes)
{
classNames[i++] = cls.getName();
}
for (String className : generatedClassNames)
{
classNames[i++] = className;
}
return classNames;
}
else if (!classes.isEmpty())
{
// Normal persistence
// Put the classes into a sorter so we make sure we get the initial implementations first followed by any subclasses of these implementations.
// This is needed because when generating the schema we require the subclass implementations to already have their datastore column created
Collection classesSorted = new TreeSet<>(new InterfaceClassComparator());
Iterator classesIter = classes.iterator();
while (classesIter.hasNext())
{
classesSorted.add(classesIter.next());
}
// Return the class names (in the same order)
String[] classNames = new String[classesSorted.size()];
Iterator iter = classesSorted.iterator();
int i = 0;
while (iter.hasNext())
{
classNames[i++] = iter.next().getName();
}
return classNames;
}
return null;
}
/**
* Simple comparator that orders the implementations of an interface so that the initial implementations are first, and the subclasses later.
*/
private static class InterfaceClassComparator implements Comparator, Serializable
{
private static final long serialVersionUID = -8114305773358090763L;
public InterfaceClassComparator()
{
// Nothing to do
}
/**
* Method defining the ordering of objects. Places all nulls at the end.
* @param cls1 First class
* @param cls2 Second class
* @return The comparison result
*/
public int compare(Class cls1, Class cls2)
{
if (cls1 == null && cls2 == null)
{
return 0;
}
else if (cls1 == null || cls2 == null)
{
return Integer.MIN_VALUE;
}
// Just order based on hashcode
return cls1.hashCode() - cls2.hashCode();
}
}
@Override
public void addORMDataToClass(Class c, ClassLoaderResolver clr)
{
// Default to doing nothing. Specified in subclasses if they support it
return;
}
@Override
public void addAnnotationsDataToClass(Class c, AbstractClassMetaData cmd, ClassLoaderResolver clr)
{
if (allowAnnotations)
{
// Get the MetaData for this class/interface
if (cmd.getPackageMetaData() != null && cmd.getPackageMetaData().getFileMetaData() != null &&
cmd.getPackageMetaData().getFileMetaData().getType() == MetaDataFileType.ANNOTATIONS)
{
// Our MetaData is derived from the Annotations so nothing to merge!
return;
}
// Find if there is any annotations metadata available
FileMetaData filemd = loadAnnotationsForClass(c, clr, false, false);
if (filemd != null)
{
AbstractClassMetaData annotCmd = filemd.getPackage(0).getClass(0);
if (annotCmd != null)
{
postProcessClassMetaData(annotCmd, clr);
// Merge the annotations MetaData into the class MetaData
MetaDataMerger.mergeClassAnnotationsData(cmd, annotCmd, this);
}
}
}
}
/* (non-Javadoc)
* @see org.datanucleus.metadata.MetaDataManager#getMetaDataForImplementationOfReference(java.lang.Class, java.lang.Object, org.datanucleus.ClassLoaderResolver)
*/
@Override
public ClassMetaData getMetaDataForImplementationOfReference(Class referenceClass, Object implValue, ClassLoaderResolver clr)
{
if (referenceClass == null || (!referenceClass.isInterface() && referenceClass != java.lang.Object.class))
{
return null;
}
// Check if this is a "persistent interface"
Object intfMetaData = getClassMetaDataForImplementationOfPersistentInterface(referenceClass.getName());
if (intfMetaData != null)
{
return (ClassMetaData)intfMetaData;
}
// Search for the class required
ClassMetaData cmd = null;
Iterator> cmdByClassEntryIter = classMetaDataByClass.entrySet().iterator();
while (cmdByClassEntryIter.hasNext())
{
Map.Entry cmdByClassEntry = cmdByClassEntryIter.next();
String className = cmdByClassEntry.getKey();
AbstractClassMetaData cmdCls = cmdByClassEntry.getValue();
if (cmdCls instanceof ClassMetaData)
{
try
{
// Check if class is implementation of "implValue" (in the case of java.lang.Object, all will be!)
// Class cls = referenceClass.getClassLoader().loadClass(class_name);
// if (referenceClass.isAssignableFrom(cls))
if (referenceClass == Object.class || clr.isAssignableFrom(referenceClass, className))
{
// Find the base class that is an implementation
cmd = (ClassMetaData) cmdCls;
if (implValue != null && cmd.getFullClassName().equals(implValue.getClass().getName()))
{
return cmd;
}
cmdCls = cmd.getSuperAbstractClassMetaData();
while (cmdCls != null)
{
// if (!referenceClass.isAssignableFrom(clr.classForName(((ClassMetaData)cmd_superclass).getFullClassName())))
if (referenceClass != Object.class && !clr.isAssignableFrom(referenceClass, ((ClassMetaData)cmdCls).getFullClassName()))
{
// No point going further up since no longer an implementation
break;
}
cmd = (ClassMetaData) cmdCls;
if (implValue != null && cmd.getFullClassName().equals(implValue.getClass().getName()))
{
break;
}
// Go to next superclass
cmdCls = cmdCls.getSuperAbstractClassMetaData();
if (cmdCls == null)
{
break;
}
}
}
}
catch (Exception e)
{
}
}
}
return cmd;
}
/* (non-Javadoc)
* @see org.datanucleus.metadata.MetaDataManager#getMetaDataForQuery(java.lang.Class, org.datanucleus.ClassLoaderResolver, java.lang.String)
*/
@Override
public QueryMetaData getMetaDataForQuery(Class cls, ClassLoaderResolver clr, String queryName)
{
if (queryName == null || queryMetaDataByName == null)
{
return null;
}
String query_key = queryName;
if (cls != null)
{
query_key = cls.getName() + "_" + queryName;
}
return queryMetaDataByName.get(query_key);
}
/* (non-Javadoc)
* @see org.datanucleus.metadata.MetaDataManager#getNamedQueryNames()
*/
@Override
public Set getNamedQueryNames()
{
if (queryMetaDataByName == null || queryMetaDataByName.isEmpty())
{
return null;
}
return queryMetaDataByName.keySet();
}
/* (non-Javadoc)
* @see org.datanucleus.metadata.MetaDataManager#getMetaDataForStoredProcQuery(java.lang.Class, org.datanucleus.ClassLoaderResolver, java.lang.String)
*/
@Override
public StoredProcQueryMetaData getMetaDataForStoredProcQuery(Class cls, ClassLoaderResolver clr, String queryName)
{
if (queryName == null || storedProcQueryMetaDataByName == null)
{
return null;
}
String query_key = queryName;
if (cls != null)
{
query_key = cls.getName() + "_" + queryName;
}
return storedProcQueryMetaDataByName.get(query_key);
}
/* (non-Javadoc)
* @see org.datanucleus.metadata.MetaDataManager#getMetaDataForFetchPlan(java.lang.String)
*/
@Override
public FetchPlanMetaData getMetaDataForFetchPlan(String name)
{
if (name == null || fetchPlanMetaDataByName == null)
{
return null;
}
return fetchPlanMetaDataByName.get(name);
}
/* (non-Javadoc)
* @see org.datanucleus.metadata.MetaDataManager#getMetaDataForSequence(org.datanucleus.ClassLoaderResolver, java.lang.String)
*/
@Override
public SequenceMetaData getMetaDataForSequence(ClassLoaderResolver clr, String seqName)
{
if (seqName == null || sequenceMetaDataByPackageSequence == null)
{
return null;
}
return sequenceMetaDataByPackageSequence.get(seqName);
}
/* (non-Javadoc)
* @see org.datanucleus.metadata.MetaDataManager#getMetaDataForTableGenerator(org.datanucleus.ClassLoaderResolver, java.lang.String)
*/
@Override
public TableGeneratorMetaData getMetaDataForTableGenerator(ClassLoaderResolver clr, String genName)
{
if (genName == null || tableGeneratorMetaDataByPackageSequence == null)
{
return null;
}
return tableGeneratorMetaDataByPackageSequence.get(genName);
}
/* (non-Javadoc)
* @see org.datanucleus.metadata.MetaDataManager#getMetaDataForQueryResult(java.lang.String)
*/
@Override
public QueryResultMetaData getMetaDataForQueryResult(String name)
{
if (name == null || queryResultMetaDataByName == null)
{
return null;
}
return queryResultMetaDataByName.get(name);
}
// ------------------------------- Persistent Interfaces ---------------------------------------
/* (non-Javadoc)
* @see org.datanucleus.metadata.MetaDataManager#getMetaDataForInterface(java.lang.Class, org.datanucleus.ClassLoaderResolver)
*/
@Override
public InterfaceMetaData getMetaDataForInterface(Class c, ClassLoaderResolver clr)
{
return null;
}
/* (non-Javadoc)
* @see org.datanucleus.metadata.MetaDataManager#isPersistentInterface(java.lang.String)
*/
@Override
public boolean isPersistentInterface(String name)
{
// Default to not supporting "persistent-interface"s
return false;
}
/* (non-Javadoc)
* @see org.datanucleus.metadata.MetaDataManager#isPersistentInterfaceImplementation(java.lang.String, java.lang.String)
*/
@Override
public boolean isPersistentInterfaceImplementation(String interfaceName, String implName)
{
// Default to not supporting "persistent-interface"s
return false;
}
/* (non-Javadoc)
* @see org.datanucleus.metadata.MetaDataManager#getImplementationNameForPersistentInterface(java.lang.String)
*/
@Override
public String getImplementationNameForPersistentInterface(String interfaceName)
{
// Default to not supporting "persistent-interface"s
return null;
}
/**
* Accessor for the metadata for the implementation of the specified "persistent-interface".
* @param interfaceName The name of the persistent interface
* @return The ClassMetaData of the implementation class
*/
protected ClassMetaData getClassMetaDataForImplementationOfPersistentInterface(String interfaceName)
{
// Default to not supporting "persistent-interface"s
return null;
}
/* (non-Javadoc)
* @see org.datanucleus.metadata.MetaDataManager#registerPersistentInterface(org.datanucleus.metadata.InterfaceMetaData, java.lang.Class, org.datanucleus.ClassLoaderResolver)
*/
@Override
public void registerPersistentInterface(InterfaceMetaData imd, Class implClass, ClassLoaderResolver clr)
{
// Default to not supporting "persistent-interface"s
return;
}
/* (non-Javadoc)
* @see org.datanucleus.metadata.MetaDataManager#registerImplementationOfAbstractClass(org.datanucleus.metadata.ClassMetaData, java.lang.Class, org.datanucleus.ClassLoaderResolver)
*/
@Override
public void registerImplementationOfAbstractClass(ClassMetaData cmd, Class implClass, ClassLoaderResolver clr)
{
// Default to not supporting "persistent-abstract-classes"
return;
}
// ------------------------------- Utilities -------------------------------
/**
* Utility to parse an XML metadata file.
* @param fileURL URL of the file
* @return The FileMetaData for this file
*/
protected abstract FileMetaData parseXmlFile(URL fileURL);
@Override
public abstract void registerFile(String fileURLString, FileMetaData filemd, ClassLoaderResolver clr);
@Override
public void registerDiscriminatorValueForClass(AbstractClassMetaData cmd, String discrimValue)
{
AbstractClassMetaData rootCmd = cmd.getBaseAbstractClassMetaData();
DiscriminatorLookup lookup = discriminatorLookupByRootClassName.get(rootCmd.getFullClassName());
if (lookup == null)
{
lookup = new DiscriminatorLookup();
discriminatorLookupByRootClassName.put(rootCmd.getFullClassName(), lookup);
}
lookup.addValue(cmd.getFullClassName(), discrimValue);
}
@Override
public String getClassNameForDiscriminatorValueWithRoot(AbstractClassMetaData rootCmd, String discrimValue)
{
DiscriminatorLookup lookup = discriminatorLookupByRootClassName.get(rootCmd.getFullClassName());
if (lookup != null)
{
return lookup.getClassForValue(discrimValue);
}
return null;
}
@Override
public String getDiscriminatorValueForClass(AbstractClassMetaData cmd)
{
AbstractClassMetaData rootCmd = cmd.getBaseAbstractClassMetaData();
DiscriminatorLookup lookup = discriminatorLookupByRootClassName.get(rootCmd.getFullClassName());
if (lookup != null)
{
return lookup.getValueForClass(cmd.getFullClassName());
}
return null;
}
/* (non-Javadoc)
* @see org.datanucleus.metadata.MetaDataManager#getClassNameFromDiscriminatorValue(java.lang.String, org.datanucleus.metadata.DiscriminatorMetaData)
*/
@Override
public String getClassNameFromDiscriminatorValue(String discrimValue, DiscriminatorMetaData dismd)
{
if (discrimValue == null)
{
return null;
}
if (dismd.getStrategy() == DiscriminatorStrategy.CLASS_NAME)
{
// TODO If classMetaData not known load it?
return discrimValue;
}
else if (dismd.getStrategy() == DiscriminatorStrategy.VALUE_MAP)
{
AbstractClassMetaData baseCmd = (AbstractClassMetaData)((InheritanceMetaData)dismd.getParent()).getParent();
AbstractClassMetaData rootCmd = baseCmd.getBaseAbstractClassMetaData();
return getClassNameForDiscriminatorValueWithRoot(rootCmd, discrimValue);
}
else if (dismd.getStrategy() == DiscriminatorStrategy.VALUE_MAP_ENTITY_NAME)
{
AbstractClassMetaData baseCmd = (AbstractClassMetaData)((InheritanceMetaData)dismd.getParent()).getParent();
AbstractClassMetaData rootCmd = baseCmd.getBaseAbstractClassMetaData();
String value = getClassNameForDiscriminatorValueWithRoot(rootCmd, discrimValue);
if (value != null)
{
return value;
}
// Fallback to the entity name
AbstractClassMetaData cmd = classMetaDataByEntityName.get(discrimValue);
return (cmd != null) ? cmd.getFullClassName() : null;
}
return null;
}
@Override
public void registerSequencesForFile(FileMetaData filemd)
{
// Register all sequences for the packages in this file
for (int i=0;i();
}
// The problem here is that with JDO we want the sequence to be fully-qualified yet JPA/Jakarta wants the sequence name itself.
// Also we could be using JPA/Jakarta annotations with JDO persistence, or mixed mode, so need to cater for both ways
for (SequenceMetaData seqmd : seqmds)
{
// Register using its fully qualified name (JDO)
sequenceMetaDataByPackageSequence.put(seqmd.getFullyQualifiedName(), seqmd);
// Register using its basic name (JPA/Jakarta)
sequenceMetaDataByPackageSequence.put(seqmd.getName(), seqmd);
}
}
}
}
@Override
public void registerTableGeneratorsForFile(FileMetaData filemd)
{
// Register all table generators for the packages in this file
for (int i=0;i();
}
// The problem here is that with JDO we want the generator to be fully-qualified
// yet JPA wants the generator name itself. Also we could be using JPA annotations
// with JDO persistence, or mixed mode, so need to cater for both ways
for (TableGeneratorMetaData tgmd : tgmds)
{
// Register using its fully qualified name (JDO)
tableGeneratorMetaDataByPackageSequence.put(tgmd.getFullyQualifiedName(), tgmd);
// Register using its basic sequence name
tableGeneratorMetaDataByPackageSequence.put(tgmd.getName(), tgmd);
}
}
}
}
/**
* Convenience method to register the query result metadata for the provided file.
* @param filemd MetaData for the file
*/
protected void registerQueryResultMetaDataForFile(FileMetaData filemd)
{
// Register all query result mappings for the file
QueryResultMetaData[] fqrmds = filemd.getQueryResultMetaData();
if (fqrmds != null)
{
if (queryResultMetaDataByName == null)
{
queryResultMetaDataByName = new ConcurrentHashMap<>();
}
for (QueryResultMetaData qrmd : fqrmds)
{
queryResultMetaDataByName.put(qrmd.getName(), qrmd);
}
}
// Register all query result mappings for the classes in the file
int numPackages = filemd.getNoOfPackages();
for (int i=0;i();
}
for (QueryResultMetaData qrmd : qrmds)
{
queryResultMetaDataByName.put(qrmd.getName(), qrmd);
}
}
}
}
}
/**
* Convenience method to register a query under a name.
* @param qmd Query metadata
*/
public void registerNamedQuery(QueryMetaData qmd)
{
if (queryMetaDataByName == null)
{
queryMetaDataByName = new ConcurrentHashMap<>();
}
String scope = qmd.getScope();
String key = qmd.getName();
if (scope != null)
{
key = scope + "_" + key;
}
queryMetaDataByName.put(key, qmd);
}
/**
* Convenience method to register all queries found in the passed file.
* @param filemd MetaData for the file
*/
protected void registerQueriesForFile(FileMetaData filemd)
{
// Register all queries for this file
// Store queries against "queryname"
QueryMetaData[] queries = filemd.getQueries();
if (queries != null)
{
if (queryMetaDataByName == null)
{
queryMetaDataByName = new ConcurrentHashMap<>();
}
for (QueryMetaData qmd : queries)
{
String scope = qmd.getScope();
String key = qmd.getName();
if (scope != null)
{
key = scope + "_" + key;
}
queryMetaDataByName.put(key, qmd);
}
}
int numPackages = filemd.getNoOfPackages();
for (int i = 0; i < numPackages; i++)
{
PackageMetaData pmd = filemd.getPackage(i);
// Register all classes (and their queries) into the respective lookup maps
int numClasses = pmd.getNoOfClasses();
for (int j = 0; j < numClasses; j++)
{
// Store queries against "classname_queryname"
ClassMetaData cmd = pmd.getClass(j);
QueryMetaData[] classQueries = cmd.getQueries();
if (classQueries != null)
{
if (queryMetaDataByName == null)
{
queryMetaDataByName = new ConcurrentHashMap<>();
}
for (QueryMetaData qmd : classQueries)
{
String scope = qmd.getScope();
String key = qmd.getName();
if (scope != null)
{
key = scope + "_" + key;
}
queryMetaDataByName.put(key, qmd);
}
}
}
// Register all interfaces (and their queries) into the respective lookup maps
int numInterfaces = pmd.getNoOfInterfaces();
for (int j = 0; j < numInterfaces; j++)
{
// Store queries against "classname_queryname"
InterfaceMetaData intfmd = pmd.getInterface(j);
QueryMetaData[] interfaceQueries = intfmd.getQueries();
if (interfaceQueries != null)
{
if (queryMetaDataByName == null)
{
queryMetaDataByName = new ConcurrentHashMap<>();
}
for (QueryMetaData qmd : interfaceQueries)
{
String scope = qmd.getScope();
String key = qmd.getName();
if (scope != null)
{
key = scope + "_" + key;
}
queryMetaDataByName.put(key, qmd);
}
}
}
}
}
/**
* Convenience method to register all stored proc queries found in the passed file.
* @param filemd MetaData for the file
*/
protected void registerStoredProcQueriesForFile(FileMetaData filemd)
{
// Register all queries for this file
// Store queries against "queryname"
StoredProcQueryMetaData[] queries = filemd.getStoredProcQueries();
if (queries != null)
{
if (storedProcQueryMetaDataByName == null)
{
storedProcQueryMetaDataByName = new ConcurrentHashMap<>();
}
for (int i=0;i();
}
for (StoredProcQueryMetaData spqmd : classStoredProcQueries)
{
String key = spqmd.getName();
storedProcQueryMetaDataByName.put(key, spqmd);
}
}
}
// Register all interfaces (and their queries) into the respective lookup maps
for (int j = 0; j < pmd.getNoOfInterfaces(); j++)
{
// Store queries against "classname_queryname"
InterfaceMetaData intfmd = pmd.getInterface(j);
StoredProcQueryMetaData[] interfaceStoredProcQueries = intfmd.getStoredProcQueries();
if (interfaceStoredProcQueries != null)
{
if (storedProcQueryMetaDataByName == null)
{
storedProcQueryMetaDataByName = new ConcurrentHashMap<>();
}
for (StoredProcQueryMetaData spqmd : interfaceStoredProcQueries)
{
String key = spqmd.getName();
storedProcQueryMetaDataByName.put(key, spqmd);
}
}
}
}
}
/**
* Convenience method to register all FetchPlans found in the passed file.
* @param filemd MetaData for the file
*/
protected void registerFetchPlansForFile(FileMetaData filemd)
{
// Register all fetch plans for this file. Store FetchPlanMetaData against fetch plan name
FetchPlanMetaData[] fetchPlans = filemd.getFetchPlans();
if (fetchPlans != null)
{
if (fetchPlanMetaDataByName == null)
{
fetchPlanMetaDataByName = new ConcurrentHashMap<>();
}
for (FetchPlanMetaData fpmd : fetchPlans)
{
fetchPlanMetaDataByName.put(fpmd.getName(), fpmd);
}
}
}
/**
* Convenience method to populate all classes/interfaces in a Meta-Data file.
* @param filemd The MetaData file
* @param clr Class Loader to use in population
* @param primary the primary ClassLoader to use (or null)
*/
protected void populateFileMetaData(FileMetaData filemd, ClassLoaderResolver clr, ClassLoader primary)
{
for (int i=0;i appIdCmds = classMetaDataByAppIdClassName.get(cmd.getObjectidClass());
if (appIdCmds == null)
{
appIdCmds = new HashSet<>();
classMetaDataByAppIdClassName.put(cmd.getObjectidClass(), appIdCmds);
appIdCmds.add(cmd);
}
else
{
appIdCmds.add(cmd);
}
}
if (cmd instanceof ClassMetaData)
{
if (cmd.getPersistableSuperclass() != null)
{
// Add to direct subclasses for the superclass
Set directSubclasses = directSubclassesByClass.get(cmd.getPersistableSuperclass());
if (directSubclasses == null)
{
directSubclasses = new HashSet();
directSubclassesByClass.put(cmd.getPersistableSuperclass(), directSubclasses);
}
directSubclasses.add(cmd.getFullClassName());
if (!((ClassMetaData)cmd).isAbstract())
{
// Concrete class, so add to cache of concrete subclasses for all superclasses
AbstractClassMetaData theCmd = cmd;
while (theCmd.getPersistableSuperclass() != null)
{
theCmd = theCmd.getSuperAbstractClassMetaData();
Set subclassNames = concreteSubclassNamesByClassName.get(theCmd.getFullClassName());
if (subclassNames == null)
{
subclassNames = new HashSet<>();
concreteSubclassNamesByClassName.put(theCmd.getFullClassName(), subclassNames);
}
subclassNames.add(cmd.getFullClassName());
}
}
}
}
if (listeners != null && listenersLoadedMetaData != null)
{
listenersLoadedMetaData.add(cmd);
}
}
/* (non-Javadoc)
* @see org.datanucleus.metadata.MetaDataManager#getConcreteSubclassesForClass(java.lang.String)
*/
@Override
public String[] getConcreteSubclassesForClass(String className)
{
Set concreteSubclasses = concreteSubclassNamesByClassName.get(className);
return concreteSubclasses == null ? null : concreteSubclasses.toArray(new String[concreteSubclasses.size()]);
}
/* (non-Javadoc)
* @see org.datanucleus.metadata.MetaDataManager#getSubclassesForClass(java.lang.String, boolean)
*/
@Override
public String[] getSubclassesForClass(String className, boolean includeDescendents)
{
Collection subclassNames2 = new HashSet<>();
provideSubclassesForClass(className, includeDescendents, subclassNames2);
if (!subclassNames2.isEmpty())
{
return subclassNames2.toArray(new String[subclassNames2.size()]);
}
return null;
}
/**
* Provide the subclasses of a particular class to a given consumer
* @param className Name of the class that we want the known subclasses for.
* @param includeDescendents Whether to include subclasses of subclasses etc
* @param consumer the Collection (Set) where discovered subclasses are added
*/
private void provideSubclassesForClass(String className, boolean includeDescendents, Collection consumer)
{
Set subclasses = directSubclassesByClass.get(className);
if (subclasses != null)
{
consumer.addAll(subclasses);
if (includeDescendents)
{
for (String subClassName : subclasses)
{
//go deeper in subclasses
provideSubclassesForClass(subClassName, includeDescendents, consumer);
}
}
}
}
/* (non-Javadoc)
* @see org.datanucleus.metadata.MetaDataManager#getReferencedClasses(java.lang.String[], org.datanucleus.ClassLoaderResolver)
*/
@Override
public List getReferencedClasses(String[] classNames, ClassLoaderResolver clr)
{
List cmds = new ArrayList<>();
for (String className : classNames)
{
Class cls = null;
try
{
cls = clr.classForName(className);
if (!cls.isInterface())
{
AbstractClassMetaData cmd = getMetaDataForClass(className, clr);
if (cmd == null)
{
NucleusLogger.DATASTORE.warn("Class Invalid " + className);
throw new NoPersistenceInformationException(className);
}
cmds.addAll(getReferencedClassMetaData(cmd, clr));
}
}
catch (ClassNotResolvedException cnre)
{
// Class not found so ignore it
NucleusLogger.DATASTORE.warn("Class " + className + " not found so being ignored");
}
}
return cmds;
}
/**
* Utility to return all ClassMetaData that is referenced from the supplier class.
* @param cmd The origin class's MetaData.
* @param clr ClassLoaderResolver resolver for loading any classes.
* @return List of ClassMetaData referenced by the origin
*/
protected List getReferencedClassMetaData(AbstractClassMetaData cmd, ClassLoaderResolver clr)
{
if (cmd == null)
{
return null;
}
// Use the ClassMetaData to tell us about its classes
List orderedCmds = new ArrayList<>();
Set referencedCmds = new HashSet<>();
cmd.getReferencedClassMetaData(orderedCmds, referencedCmds, clr);
return orderedCmds;
}
/* (non-Javadoc)
* @see org.datanucleus.metadata.MetaDataManager#isFieldTypePersistable(java.lang.Class)
*/
@Override
public boolean isFieldTypePersistable(Class type)
{
if (isEnhancing())
{
// Enhancing so return if we have MetaData that is persistable
AbstractClassMetaData cmd = readMetaDataForClass(type.getName());
if (cmd != null && cmd instanceof ClassMetaData && cmd.getPersistenceModifier() == ClassPersistenceModifier.PERSISTENCE_CAPABLE)
{
return true;
}
}
return getApiAdapter().isPersistable(type);
}
}