/**********************************************************************
Copyright (c) 2006 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.jpox.store.rdbms.table;
import java.util.ArrayList;
import java.util.Collection;
import java.util.List;
import org.jpox.ClassLoaderResolver;
import org.jpox.metadata.AbstractClassMetaData;
import org.jpox.metadata.AbstractMemberMetaData;
import org.jpox.metadata.ForeignKeyAction;
import org.jpox.metadata.ForeignKeyMetaData;
import org.jpox.metadata.IndexMetaData;
import org.jpox.metadata.UniqueMetaData;
import org.jpox.store.mapped.DatastoreClass;
import org.jpox.store.mapped.DatastoreContainerObject;
import org.jpox.store.mapped.DatastoreIdentifier;
import org.jpox.store.mapped.IdentifierFactory;
import org.jpox.store.mapped.MappedStoreManager;
import org.jpox.store.mapped.mapping.JavaTypeMapping;
import org.jpox.store.mapped.mapping.ReferenceMapping;
import org.jpox.store.rdbms.key.CandidateKey;
import org.jpox.store.rdbms.key.ForeignKey;
import org.jpox.store.rdbms.key.Index;
/**
* Class containing a series of convenience methods for the generation of tables and constraints.
*
* @version $Revision: 1.11 $
*/
public class TableUtils
{
/**
* Convenience method to add foreign-keys for the specified reference field.
* Adds FKs from the column(s) in this table to the ID column(s) of the PC table of the implementation type.
* @param fieldMapping The field mapping (in this table)
* @param fmd MetaData for this field
* @param autoMode Whether we are in auto-create mode
* @param storeMgr Store Manager
* @param clr ClassLoader resolver
* @return The foreign key(s) created
*/
public static Collection getForeignKeysForReferenceField(JavaTypeMapping fieldMapping,
AbstractMemberMetaData fmd,
boolean autoMode,
MappedStoreManager storeMgr,
ClassLoaderResolver clr)
{
final ReferenceMapping refMapping = (ReferenceMapping)fieldMapping;
JavaTypeMapping[] refJavaTypeMappings = refMapping.getJavaTypeMapping();
List fks = new ArrayList();
for (int i=0;i<refJavaTypeMappings.length;i++)
{
// If the implementation is of a PC class, look to add a FK to the PC class table
final JavaTypeMapping implMapping = refJavaTypeMappings[i];
if (storeMgr.getOMFContext().getMetaDataManager().getMetaDataForClass(implMapping.getType(), clr) != null &&
implMapping.getNumberOfDatastoreFields() > 0)
{
DatastoreClass referencedTable = storeMgr.getDatastoreClass(implMapping.getType(), clr);
if (referencedTable != null)
{
ForeignKeyMetaData fkmd = fmd.getForeignKeyMetaData();
if ((fkmd != null && fkmd.getDeleteAction() != ForeignKeyAction.NONE) || autoMode)
{
// Either has been specified by user, or using autoMode, so add FK
ForeignKey fk = new ForeignKey(implMapping, storeMgr.getDatastoreAdapter(), referencedTable, true);
fk.setForMetaData(fkmd); // Does nothing when no FK MetaData
fks.add(fk);
}
}
}
}
return fks;
}
/**
* Convenience method to add a foreign key for a PC field.
* Adds a FK from the PC column(s) in this table to the ID columns in the PC's table.
* @param fieldMapping Mapping for the PC field
* @param fmd MetaData for the field
* @param autoMode Whether we are in auto-create mode
* @param storeMgr Store Manager
* @param clr ClassLoader resolver
* @return The ForeignKey (if any)
*/
public static ForeignKey getForeignKeyForPCField(JavaTypeMapping fieldMapping,
AbstractMemberMetaData fmd,
boolean autoMode,
MappedStoreManager storeMgr,
ClassLoaderResolver clr)
{
DatastoreClass referencedTable = storeMgr.getDatastoreClass(fmd.getTypeName(), clr);
if (referencedTable == null)
{
// PC type uses subclass-table
AbstractClassMetaData refCmd = storeMgr.getOMFContext().getMetaDataManager().getMetaDataForClass(fmd.getType(), clr);
if (refCmd != null)
{
AbstractClassMetaData[] refCmds = storeMgr.getClassesManagingTableForClass(refCmd, clr);
if (refCmds != null && refCmds.length == 1)
{
referencedTable = storeMgr.getDatastoreClass(refCmds[0].getFullClassName(), clr);
}
else
{
// "subclass-table" with more than 1 subclass with table (not supported)
}
}
}
if (referencedTable != null)
{
ForeignKeyMetaData fkmd = fmd.getForeignKeyMetaData();
if ((fkmd != null && fkmd.getDeleteAction() != ForeignKeyAction.NONE) || autoMode)
{
// Either has been specified by user, or using autoMode, so add FK
ForeignKey fk = new ForeignKey(fieldMapping, storeMgr.getDatastoreAdapter(), referencedTable, true);
fk.setForMetaData(fkmd); // Does nothing when no FK MetaData
return fk;
}
}
return null;
}
/**
* Convenience method to create an Index for a field.
* @param table Container for the index
* @param imd The Index MetaData
* @param fieldMapping Mapping for the field
* @return The Index
*/
public static Index getIndexForField(DatastoreContainerObject table, IndexMetaData imd, JavaTypeMapping fieldMapping)
{
if (fieldMapping.getNumberOfDatastoreFields() == 0)
{
// No columns in this mapping so we can hardly index it!
return null;
}
// Verify if a unique index is needed
boolean unique=false;
if (imd.isUnique() != null && imd.isUnique().booleanValue())
{
unique = true;
}
Index index = new Index(table, unique, imd.getValueForExtension("extended-setting"));
// Set the index name if required
if (imd.getName() != null)
{
IdentifierFactory idFactory = table.getStoreManager().getIdentifierFactory();
DatastoreIdentifier idxId = idFactory.newIdentifier(IdentifierFactory.INDEX, imd.getName());
index.setName(idxId.toString());
}
// Field-level index so use all columns for the field
int countFields = fieldMapping.getNumberOfDatastoreFields();
for (int j=0; j<countFields; j++)
{
index.addDatastoreField(fieldMapping.getDataStoreMapping(j).getDatastoreField());
}
return index;
}
/**
* Convenience method to return the candidate key (if any) for a field.
* @param umd The Unique MetaData
* @param fieldMapping Mapping for the field
* @return The Candidate Key
*/
public static CandidateKey getCandidateKeyForField(DatastoreContainerObject table, UniqueMetaData umd, JavaTypeMapping fieldMapping)
{
CandidateKey ck = new CandidateKey(table);
// Set the key name if required
if (umd.getName() != null)
{
IdentifierFactory idFactory = table.getStoreManager().getIdentifierFactory();
DatastoreIdentifier ckId = idFactory.newIdentifier(IdentifierFactory.CANDIDATE_KEY, umd.getName());
ck.setName(ckId.toString());
}
// Field-level index so use all columns for the field
int countFields = fieldMapping.getNumberOfDatastoreFields();
for (int j=0; j<countFields; j++)
{
ck.addDatastoreField(fieldMapping.getDataStoreMapping(j).getDatastoreField());
}
return ck;
}
}